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Prólogo 


Este libro es un texto completo para aprender a programar 
con el Z80, accesible a cualquiera aunque nunca haya escrito 
ningún programa, y útil, naturalmente, para quien trabaje con 
un 780. 

Quien ya sepa programar aprenderá aqui técnicas específicas 
que aprovechan las peculiaridades del 780 o que derivan de 
ellas. El texto cubre las técnicas elementales e intermedias 
necesarias para empezar a programar. 

La finalidad del libro es proporcionar un nivel realmente 
competente al lector que desee programar el microprocesador. 
Naturalmente, es imposible aprender a programar sólo con un 
libro, sin practicar, pero cabe esperar que el texto estimulará al 
lector hasta el punto de hacerle sentirse capaz de empezar a 
escribir programas y de resolver con un microordenador pro- 
blemas de programación sencillos y hasta moderadamente com- 
plejos. 

El libro parte de la experiencia del autor, que ha enseñado a 
programar microordenadores a más de mil alumnos, y por eso 
está altamente estructurado. Los capitulos van, por lo general, 
de lo simple a lo complejo. El lector que tenga ya conocimien- 
tos elementales de programación podrá saltarse el primer ca- 
pítulo. Quienes, por el contrario, nunca hayan escrito un pro- 
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grama, quizá necesiten leer más de una vez las secciones 
finales de algunos capítulos. El texto llevará al estudiante a 
través de todos los conceptos y técnicas básicos necesarios para 
crear programas cada vez más complicados; por ello es muy 
recomendable que se respete el,orden de los capítulos. Además, 
quien desee obtener de verdad resultados tangibles deberá 
esforzarse por resolver el mayor número posible de ejercicios. 
La dificultad de éstos se ha escalonado muy cuidadosamente, y 
todos están pensados para comprobar si el material propuesto 
se ha entendido por completo. Quien no realice los ejercicios no 
podrá aprovechar por completo el valor educativo de este libro. 
Varios de ellos, como el de multiplicación, resultarán laborio- 
sos, pero al hacerlos se aprende mediante la práctica, que es la 
única manera de aprender a programar. 

Para los que con este libro se aficionen a programar se está 
preparando otro titulado Aplicaciones del Z80, que le servirá de 
complemento. 

En esta misma colección hay otros titulos que enseñan a 
programar otros microprocesadores diferentes, 

Los verdaderamente interesados en el estudio del soporte 
físico deberían consultar los títulos From Chips to Systems: an 
Introduction to Microprocessors y Microprocessor Interfacing 
Techniques. - 

El contenido de este libro se ha verificado con la mayor 
atención, y puede considerarse de fiar, aunque, inevitablemente, 
se habrán deslizado errores tipográficos o de otra clase; a este 
respecto, el autor agradecerá cualquier observación que pueda 
ser útil para lectores de futuras ediciones. Se tendrán, igualmen- 
te, en consideración cualesquiera otras sugerencias de posit'es 
mejoras, como programas deseados, desarrollados o considera- 
dos de valor por los lectores. 
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onceptos básicos 


Introducción 


En este capitulo presentaremos las ideas y definiciones 
básicas de programación de ordenadores. El lector familiarizado 
con estos temas quizá prefiera echar una ojeada rápida al 
contenido de estas páginas y pasar rápidamente al capítulo 2. 
Sin embargo, es aconsejable que incluso quien ya tenga expe- 
riencia lea esta introducción, porque en ella veremos conceptos 
muy importantes, como los de complemento a dos y representa- 
ciones BCD y de otro tipo. Algunos de ellos resultarán nuevos 
para el lector y, en cualquier caso, contribuirán a mejorar el 
nivel de conocimientos de los programadores con experiencia. 


¿Qué es programar? 


Ante un problema, lo primero que debe hacerse es idear una 
solución. À la expresión de ésta mediante una cadena de pasos 
sucesivos se le llama algoritmo. Un algoritmo es, pues, la 
especificación paso a paso de la solución de un problema. El 
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algoritmo debe terminar tras un número finito de pasos, y 
puede expresarse en cualquier lenguaje o conjunto simbólico. 
Un ejemplo de algoritmo sencillo sería el siguiente: 


Meter la llave en la cerradura. 

Dar a la llave una vuelta completa hacia la izquierda. 
Agarrar el picaporte. 

Girar el picaporte hacia la izquierda y empujar la 
puerta. 


SES 


En este punto, si el algoritmo es el adecuado a la cerradura 
en cuestión, la puerta se abrirá. Esta serie de instrucciones en 
cuatro pasos constituye un algoritmo de apertura de una 
puerta. 

Una vez que la solución a un problema se ha expresado en 
forma de algoritmo, éste debe ejecutarse en un ordenador. Por 
desgracia, es cosa sabida que los ordenadores no entienden 
español, ni cualquier otro lenguaje humano, debido a la ambi- 
güedad sintáctica inherente a todos ellos. Lo único que el 
ordenador puede entender es un subconjunto claramente defini- 
do de un lenguaje humano, y a ese subconjunto se le llama 
lenguaje de programación. 

La operación de transformar un algoritmo en una secuencia 
de instrucciones escritas en lenguaje de programación se llama 
programar. En términos estrictos, la traducción del algoritmo a 
lenguaje de programación debería llamarse codificación, puesto 
que la programación abarca también el diseño de los progra- 
mas y las “estructuras de datos” que constituirán el algoritmo. 

Para programar con eficacia hace falta no sólo conocer las 
técnicas de ejecución de algoritmos, sino también dominar 
todos los recursos que ofrece el soporte físico del ordenador 
—registros internos, memoria y dispositivos periféricos— y utilizar 
de forma creativa las estructuras de datos apropiadas. La 
descripción de estas técnicas será el objeto de los próximos 
capitulos. 

La programación obliga también a observar una estricta 
disciplina de documentación para que los programas realizados 
por una persona puedan ser entendidos por otras (y por el 
autor, pasado cierto tiempo) La documentación ha de ser 
interna y externa al programa. 

Se llama documentación interna al conjunto de comentarios 
incluidos en el propio cuerpo de un programa y que explican su 
funcionamiento. | | 

Por documentación externa se entiende la serie de explica- 
ciones escritas, manuales y diagramas de flujo escritos con 
independencia del programa. 


Diagramas de flujo 


Figura 1.1 

Diagrama de flujo de un siste- 
ma encargado de mantener 
constante la temperatura de 
una habitación. 


Entre la realización del algoritmo y la del programa se 
intercala casi siempre un diagrama de flujo, que no es sino una 
representación simbólica del algoritmo por medio de una se- 
cuencia de rectángulos y rombos que contienen los pasos del 
mismo. En los rectángulos se escriben las órdenes o “instruccio- 
nes ejecutables”. Los rombos encierran pruebas condicionales del 
tipo “si la información X es cierta, emprender la acción A, y la 
B en caso contrario”. La definición formal y la discusión de los 
diagramas de flujo se harán más adelante, al tratar de los 
programas. 


INICIO 


LEER EL VALOR "T" DE LA TEMPERATURA 
EN LA CAJA DEL TERMOSTATO 


LEER EL VALOR REAL "R" DE LA 
TEMPERATURA DE LA HABITACION EN EL 
TERMOMETRO O EN OTRO SENSOR 


(HABITACION 
MUY FRIA) MUY CALIENTE) 


CONECTAR DESCONECTAR 
EL CALENTADOR EL CALENTADOR 


(RETRASO OPCIONAL) (RETRASO OPCIONAL) 


(HABITACION 


En cualquier caso, el diagrama de flujo es un paso interme- 
dio entre la especificación del algoritmo y la codificación muy 
recomendable. A este respecto, conviene señalar que se ha 
observado que aproximadamente el 10 por 100 de los progra- 
madores son capaces de escribir buenos programas sin recurrir 
al diagrama de flujo; ¡lo malo es que también se ha observado 
que el 90 por 100 de los programadores creen pertenecer a ese 
10 por 100! Consecuencia, alrededor del 80 por 100 de los 
programas escritos por ese 90 por 100 fallan la primera vez que 
se pasan por el ordenador; como es natural, no se trata de 
porcentajes exactos. En concreto, son pocos los programadores 
noveles que comprenden la utilidad del diagrama de flujo y 
muchos los que escriben programas “sucios” o erróneos, lo que 
les obliga a perder mucho tiempo haciendo pruebas y corrigien- 
do (es lo que se llama puesta a punto). Por tanto, en todos los 
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casos es muy recomendable hacer un diagrama de flujo. Es una 
operación que lleva muy poco tiempo, pero que casi siempre 
propicia la redacción de un programa limpio que se ejecuta 
correcta y rápidamente. Hay programadores que, una vez domi- 
nada la técnica de trazado del diagrama de flujo, son capaces de 
visualizarlo mentalmente, sin necesidad de dibujarlo, aunque a 
costa de que los programas que escriben, al carecer de la 
documentación que constituye el propio diagrama de flujo, sólo 
resultan comprensibles para ellos. En resumen, siempre que se 
redacte un programa de cierta importancia, conviene acostum- 
brarse a la disciplina de diseñar el correspondiente diagrama de 
flujo. A lo largo de este libro veremos numerosos ejemplos de 
tales diagramas. 


Representación de la información 
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Todos los ordenadores manipulan información organizada 
en forma de números o de caracteres. Pasaremos a continua- 
ción a examinar las formas de representación externa e interna 
de la información en el ordenador. 


REPRESENTACION INTERNA DE LA INFORMACION 


Toda la información contenida en un ordenador se almace- 
na en forma de grupos de bits (bit es contracción de digito 
binario, es decir, “0” o “1”. Las limitaciones de la electrónica 
convencional imponen una sola forma práctica de representa- 
ción de la información: la lógica de dos estados, “0” y “1”. Un 
circuito electrónico digital conoce habitualmente dos estados: 
conectado y desconectado, estados que corresponden a las 
representaciones lógicas “0” y “1”. Dado que esos circuitos se 
utilizan para ejecutar funciones "lógicas", se les llama de "lógica 
binaria". Lo importante es que en la actualidad prácticamente 
todo el proceso de datos se lleva a cabo en formato binario. En 
el caso de los microprocesadores en general, y del Z80 en 
particular, los bits se organizan en grupos de ocho, conocidos 
por octetos y, más frecuentemente, por bytes. El conjunto de 
cuatro bits se llama nibble. 

Veamos ahora de qué forma se representa internamente este 
formato binario. Dentro del ordenador hay que representar dos 
entidades; la primera es el programa o secuencia de instruccio- 
nes; la segunda es el conjunto de datos con los que trabajará el 
programa, que pueden ser de carácter numérico o alfanumérico. 
Examinaremos a continuación la representación del programa y 
de las dos categorías de datos mencionadas. 


ВЕРКЕЗЕМТАСЮМ DEL PROGRAMA 


Todas las instrucciones se representan internamente en for- 
ma de bytes o múltiplos de bytes. Lo que se llama “instruccion 
breve” queda representada por un solo byte, mientras que las 
instrucciones más largas ocupan dos o más. Como el Z80 es un 
microprocesador de ocho bits, extrae los bytes de la memoria 
uno tras otro, lo que significa que las instrucciones de un solo 
byte se ejecutan, en principio, más rápidamente que las de 
varios. Más adelante veremos la importancia que tiene este 
aspecto del juego de instrucciones de cualquier microprocesa- 
dor, y en concreto del Z80, en el que se quiso proporcionar la 
mayor cantidad posible de instrucciones breves para optimizar 
la eficacia del programa. No obstante, la limitación de la 
longitud a ocho bits plantea restricciones notables, que expon- 
dremos en su momento. Este es un ejemplo clásico de compro- 
miso entre velocidad y flexibilidad en programación. El código 
binario en el que se representan las instrucciones lo determina 
el fabricante. El Z80, como cualquier otro microprocesador, sale 
de fábrica provisto de un juego de instrucciones fijo. Dichas 
instrucciones, que determina el fabricante, se recogen al final del 
libro junto con su código. Los programas se expresan todos 
como secuencia de esas instrucciones binarias, que para el Z80 
veremos en el capítulo 4. 


REPRESENTACION DE DATOS NUMERICOS 


Representar números no es operación sencilla, y es preciso 
diferenciar varios casos. Empezaremos por representar enteros, 
pasaremos a continuación a enteros con signo —-positivos y 
negativos— y terminaremos con la representación de números 
decimales, 

Para representar enteros puede recurrirse a la forma binaria 
directa, que no es sino la representación del valor decimal del 
nümero en sistema binario. En éste, el bit situado en el extremo 
derecho es igual a 2 elevado a la potencia 0; el situado a su 
izquierda equivale a 2 elevado a 1; el siguiente, a 2 elevado a 2, 
y el del extremo izquierdo vale 2 elevado a 7, igual a 128. 


b;bob;b,b;b;b, 6, 
representa 
b,2? + 52° + 62° + b424 + b,2? + b,2? + 6,21 + b,2? 


Las potencias de 2 son: 


27 = 128,26 = 64,25 = 32,24 = 16,2% = 8,2? = 4,21 2 2,20 = 1 
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La representación binaria es análoga a la decimal. En ésta, 
“123” equivale a: 


1 x 100 = 100 
+2x 10= 20 
+3х 12 3 


= 123 


Obsérvese que 100 = 102, 10 = 10', 1 = 10°. 
En esta “notación posicional”, cada cifra representa una 


potencia de 10. En el sistema binario, cada cifra binaria o bit 
representa una potencia de 2, 


Ejemplo: en el sistema binario, “00001001” equivale a: 


` 
oococoroo- 
хххххххх 
оо À © с со» го ~ 
CS | 
o|ocooooocoo- 


En 
NAU = 


en decimal: 


Veamos algunos otros ejemplos: “10000001” equivale a: 


How nu y m m w og 


Һ-Һооосоосо»- 


хххххххх 


— 
%o0oocoooo- 


N ON со къ 

oo Б D © © B D + 
ке| — 

N 


en decimal: 


por tanto, “10000001” equivale al número decimal 129. 

Al examinar la representación binaria de los números se 
entiende por qué los bits se numeran de O a 7 empezando por 
la derecha. El bit O es “by” y corresponde a 2°; el bit 1 es “b,” y 
corresponde a 2!, y así sucesivamente. 

La figura 1.2 recoge los binarios correspondientes a los 
números decimales comprendidos entre 0 у 255. 


Decimal Binario Decimal 


00000000 32 | 00100000 
00000001 00100001 
00000010 
00000011 
00000100 
00000101 00111111 
00000110 01000000 
00000111 01000001 
00001000 
00001001 
00001010 01111111 
00001011 10000000 
00001100 10000001 
00001101 
00001110 
00001111 
00010000 
00010001 


© 00 -1 В «лом. Фо коз © 


11111110 


Figura 1.2 : 00011111 11111111 


Tabla de equivalencias entre los 
sistemas decimal y binario. 


Ejercicio 1.1: ¿Cuál es el valor decimal de “11111100”? 


Transformación de decimal a binario 


Calcúlese el equivalente binario del número decimal “11”: 


11-2=5 resto 11 (LSB) 
5:2=2 resto 1 > 1 
222 = 1 resto 0-0 
1-2 = 0 resto 151 (MSB) 


El binario equivalente se obtiene leyendo la columna de la 
derecha desde abajo hacia arriba: 1011. 

El equivalente binario de un decimal se calcula dividiéndolo 
sucesivamente por 2 hasta obtener un cociente nulo. 


Ejercicio 1.2: ¿Cuál es el número binario equivalente al decimal 
257? 


Ejercicio 1.3: Transfórmese 19 a notación binaria, y de nuevo 
a decimal. 
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Operaciones con datos binarios 


Las reglas aritméticas son directas y sencillas. La suma, por 
ejemplo, es: 


[oH d gd 


-- о Ф 
= 
— 


++++ 
— о – Ф 
о — ~ о 


donde (1) significa el acarreo de 1 (“llevarse” 1; obsérvese que 
“10” en el sistema binario equivale al decimal “2”). La sustrac- 
ción binaria se realiza sumando el complemento, y se explicará 
al tratar de la representación de números negativos. 


Ejemplo: 
(2) 10 
+0 +01 
= (3) 11 


La suma se realiza igual que en base decimal, sumando las 
columnas una por una, a partir de la primera por la derecha: 
Suma de la columna derecha: 


10 
+ 01 


(0 +1 = 1. No se lleva nada) 


Suma de la siguiente columna: 


01 
КЕ (1+0 = 1. № se lleva nada) 


Ejercicio 1.4: Calcúlese 5 + 10 en el sistema binario, comproban- 
do si el resultado es efectivamente 15. 


Algunos otros ejemplos de adición binaria: 


0010 (2) 0011 (3) 
+ 0001 (1) + 0001 (1) 
= 0011 (3) = 0100 (4) 


Este último ejemplo ilustra la función del acarreo. 
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En efecto, Щетопоз en la columna de la derecha: 1 + 1 
= (1)0; una vez efectuada la suma. llevamos 1. que se suma a 
la siguiente columna: 


001 la columna 0 acaba de sumarse 
+ 000 
+ 1 (acarreo) 


= (1)0 siendo (1) el nuevo acarreo 
a la columna 2. 


El resultado final es 0100. 
Otro ejemplo: 


0111 (7) 
+ 0011 + (3) 
1010 = (10) 


En este ejemplo vuelve a generarse un acarreo, que se lleva 
hasta la columna de la izquierda. 


Ejercicio 1.5: Calcúlese la suma: 


1111 
+0001 
=? 


¿Cabe el resultado en cuatro bits? 


Por tanto, con ocho bits pueden representarse directamente 
los números comprendidos entre “00000000” y “11111111”, es 
decir, entre “0” y “255”. Inmediatamente se plantean dos dificul- 
tades: primera, que sólo representamos números positivos; 
segunda, que la magnitud de esos números queda limitada a 
255, si trabajamos con sólo ocho bits. Veamos de qué forma se 
resuelven. 


Binario con signo 


En un nümero representado en notación binaria con signo. 
éste viene indicado por el bit de la izquierda, tradicionalmente 
“0”, si es positivo, y “1”, si es negativo; por tanto, “11111111” 
representa — 127, y “01111111”, + 127. De esta forma ya 
podemos representar nümeros positivos y negativos, pero a 
costa de reducir la magnitud de 255 a 127. 
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Ejemplo: “0000 0001” equivale a + 1 (el primer “0” es “+”; 
el resto, “0000001”, es igual a 1). 
“1000 0001” es — 1 (el primer “1” es “—”). 


ec 


Ejercicio 1.6: ¿Cómo se representaría " — 5" en notación binaria 


con signo? 


Pasemos ahora al problema de la magnitud. Para represen- 
tar nümeros más grandes no hay más remedio que utilizar 
mayor nümero de bits. Así, con dieciséis bits (dos bytes) pode- 
mos representar todos los números comprendidos entre — 32K 
y + 32K (en lenguaje informático, ІК es igual а 1 024). El bit 
15 lleva el signo, y los quince restantes (los comprendidos entre 
el O y el 14) expresan la magnitud: 21° = 32K. Si esta magnitud 
sigue siendo pequeña, no hay más que usar tres bytes o más. 
En resumen, cuanto más grande sea la magnitud del entero que 
queramos representar, tanto mayor será el nümero de bytes 
necesario para ello. Por eso, las versiones más sencillas del 
BASIC y otros lenguajes disponen de una precisión limitada 
para representar enteros, ya que necesitan manipular interna- 
mente las cantidades en un formato más corto. Las mejores 
versiones del BASIC y de los demás lenguajes ofrecen más cifras 
decimales significativas, pero a costa de reservar más bytes para 
cada nümero. 

Otro extremo que debemos considerar es el de la velocidad. 
Veamos, por ejemplo, cómo se lleva a cabo la adición de dos 
nümeros en la representación binaria con signo que acabamos 
de estudiar. Sea la suma de “— 5" y “+ 7": 


+ 7 se representa como 00000111 
— 5 se representa como 10000101 


la suma binaria es 10001100, o — 12 


Pero el resultado correcto no es - 12, sino + 2. Para 
trabajar en esta representación hay que atenerse a ciertas reglas 
determinadas, que dependen del signo. La consecuencia es que 
aumenta la complejidad y disminuye la eficacia. En otras 
palabras, la adición binaria de números con signo “no marcha 
bien”. La situación es, por tanto, delicada, porque el ordenador, 
además de representar información, tiene que ejecutar con ella 
operaciones aritméticas. 

La solución al problema viene dada por lo que se llama 
representación en complemento a dos, que sustituye a la binaria 
con signo. En lugar de abordarla directamente, nos detendremos 
antes un poco en el complemento a uno. ` 


Complemento a uno 


En complemento a uno todos los enteros positivos se repre-. 


sentan en su formato binario correcto. Аз, “+ 3" se representa 
como 00000011; por el contrario, “— 3” se representa determi- 
nando el complemento de cada uno de los bits de la representa- 
ción original, lo que equivale a transformar todos los O en 1 y 
todos los 1 en 0. En el ejemplo que nos ocupa, la representa- 
ción de “— 3” en complemento a uno sería 11111100. 

Otro ejemplo: 


+ 2 es 00000010 
— 2 es 11111101 


Obsérvese que en esta representación el primer bit de la iz- 
quierda es “0”, si el número es positivo, y “1”, si es negativo. 


Ejercicio 1.7: La representación de “+ 6” es 00000110". ¿Cuál 
será la de “-- 6" en complemento a uno? 


Probemos a sumar ahora — 4 y +6: 


— 4 es 11111011 
+ 6 es 00000110 


la suma es: A 00000001, donde (1) indica un acarreo. 


El “resultado correcto” será, por tanto, “2”, o bien 
“00000010”. 
Veamos otro caso: 


— 3 es 11111100 
— 2 es 11111101 


la suma es: (1) 11111001 


o bien “— 6" más una unidad que se acarrea; pero, en realidad, 
debería ser “— 5”, es decir, 11111010, lo que quiere decir que el 
procedimiento no funciona. 

Este formato sirve para representar números positivos y 
negativos, pero el resultado de la adición no siempre es correc- 
to. Es preciso, por tanto, idear otra representación, que en este 
caso será el complemento a dos, evolución del complemento a 
uno. 


Representación en complemento a dos 


Los números positivos se representan como binarios con 
signo, exactamente igual que se hacia en complemento a uno. 


23 
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La diferencia estriba en la representación de los números negati- 
vos, que se hace determinando primero el complemento a uno y 
sumando uno a continuación. ` 

Veámoslo en un ejemplo: + 3 se representa como binario 
con signo por 00000011; su representación en complemento a 
uno es 11111100. El complemento a dos se obtiene sumando 1 a 
esta última, lo que da 11111101. 

Apliquemos el procedimiento a la adición: 


3) 0000001 1 
+ (5) + 00000101 
= (8) = 00001000 


el resultado es correcto. 
Veamos ahora qué ocurre en la sustracción: 


(3) 00000011 
(— 5) + 11111011 
= 11111110 


Para identificar el resultado, calcularemos el complemento a 
dos: 


el complemento a dos de 11111110 es 00000001 
se suma ] + 1 


el complemento a dos es, pues 00000010, o + 2 


El resultado anterior — “11111110” representa “—2” y es, por 
tanto, correcto. 

Los resultados —ignorando el arrastre— han sido correctos 
tanto en la adición como en la sustracción, lo que parece 
indicar que el complemento a dos funciona. 


Ejercicio 1.8: ¿Cuál es la representación de “+ 127” en comple- 
mento a dos? 


Ejercicio 1.9: ¿Cuál es la representación de “— 128" en comple- 
mento a dos? 


Probemos ahora a sumar + 4 y — 3 (la sustracción se realiza 
sumando el complemento a dos): 


+ 4 es 00000100 


— 3 es 11111101 
€l resultado es: (1) 00000001 


Si ignoramos el acarreo, el resultado es 00000001, es decir, 
“1? en representación decimal, es, por tanto, correcto. Aunque 
no daremos la demostración matemática completa, digamos por 
el momento que esta representación funciona y que en comple- 
mento a dos se pueden sumar y restar números con signo con 
independencia del mismo. Al aplicar la regla normal de la 
adición en binario se obtiene el resultado correcto incluyendo el 
signo. El arrastre se ignora, lo que constituye una ventaja 
considerable, porque en caso contrario sería preciso corregir 
siempre el signo en el resultado, con el consiguiente alargamien- 
to del tiempo de operación. 

Para resumir, digamos que el complemento a dos constituye 
la forma de representación más adecuada para los procesadores 
más simples, como los microprocesadores. En procesadores 
complejos puede recurrirse a otras formas de representación, 
como el complemento a uno, usando un circuito especial para 
corregir el resultado. 

А partir de este punto, todos los enteros con signo зе 
supondrán representados internamente en notación de comple- 
mento a dos. La tabla de la figura 1.3 recoge los complementos 
a dos de varios números. 


Ejercicio 1.10: ¿Cuáles son los números máximo y mínimo que 
6 
pueden representarse en complemento а dos con sólo ип byte? 


Ejercicio 1.11: Calcúlese el complemento a dos de 20 y a conti- 
nuación el del resultado obtenido. ¿Se vuelve a obtener 20? 


Veamos a continuación, mediante algunos ejemplos, la for- 
ma en que se aplica el complemento a dos. Llamaremos C al 
acarreo, que corresponde al bit 8 del resultado. 

V indica desbordamiento o cambio de signo “accidental” a 
consecuencia de la excesiva magnitud de los números con que 
se opera. Se trata básicamente de un acarreo interno del bit 6 
al bit 7 (el bit de signo), y examinaremos sus consecuencias a 
continuación. 


Acarreo С 


He aquí un ejemplo de acarreo: 


(128) 10000000 
+ (129) + 10000001 


(257) = (1) 00000001 


siendo (1) el acarreo. 
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Código en Código en 
complemento a dos complemento a dos 


01111111 

01111110 кү 

01111101 10000010 
10000011 


01000001 10111111 
01000000 11000000 
00111111 11000001 


00100001 11011111 
00100000 11100000 
00011111 11100001 


00010001 11101111 


00010000 11110000 
00001111 11110001 
00001110 ш 11110010 
00001101 11110011 
00001100 11110100 
00001011 11110101 
00001010 11110110 
00001001 11110111 
00001000 11111000 
00000111 11111001 
00000110 11111010 
00000101 11111011 
00000100 11111100 
0000001 1 11111101 
00000010 11111110 
00000001 11111111 
00000000 


Figura 1.3 
Tabla de complementos a dos. 


El resultado exige el uso de un noveno bit o bit 8 (ya que la 
cuenta empieza por el bit 0), llamado bit de arrastre. 

Si suponemos que el acarreo es el noveno bit del resultado, 
vemos que éste es, efectivamente, 100000001 = 257. 

El acarreo debe detectarse y manipularse con atención. En 
el interior del microprocesador, los registros encargados de 
almacenar la información tienen, por lo general, una amplitud 
de sólo ocho bits; en este ejemplo sólo se registrarían los bits 
Оа 7. 
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Por tanto, el acarreo exige siempre un cuidado especial, y 
debe detectarse y procesarse mediante instrucciones determina- 
das. El proceso sigue una de estas tres alternativas: almacenar 
el acarreo en algún otro sitio (mediante una instrucción espe- 
cial), ignorarlo o, si el mayor resultado permitido es 
“11111111”, considerarlo un error. 


Desbordamiento O 
Veamos un ejemplo de desbordamiento: 

bit 6 

bit 7 


01000000 (64) 
+ 01000001 + (65) 


= 10000001 = (— 127) 


Como se observa, se ha producido un acarreo interno del bit 
6 al bit 7; es lo que se llama desbordamiento. 

Debido a ese “accidente”, el resultado es negativo; es, pues, 
preciso detectar la situación, para corregirla. 

Examinemos otro caso: 


пий (1) 
xdi +(—1) 


= (1) 11111110 -(-2) 
Y 


acarreo 


En este caso se ha producido un acarreo interno del bit 6 al 
bit 7, y de éste al bit 8 (el acarreo C que analizamos en la 
sección anterior). Como las reglas del complemento a dos 
especifican que dicho acarreo debe ignorarse, el resultado 
obtenido es el correcto. 

Estrictamente hablando, esta última no es una situación de 
desbordamiento, ya que el acarreo del bit 6 al bit 7 no ha 
tenido como consecuencia el cambio de signo. 

Al trabajar con números negativos, el desbordamiento no se 
limita a un acarreo del bit 6 al bit 7. 

Veamos un nuevo ejemplo: 


11000000 (— 64) 
+ 10111111 — (— 65) 


= (1) 01111111. (+ 127) 
Y 


acarreo 


2 
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Esta vez no ha habido acarreo interno del bit 6 al bit 7, 
sino acarreo externo, pero el resultado es, no obstante, inco- 
rrecto, porque ha cambiado el bit 7; se trata de una situación 
de desbordamiento. 

En resumen, se producirá desbordamiento en cuatro casos: 


1. Adición de números positivos muy grandes. 

2. Adición de números negativos muy grandes. 

3. Sustracción de un positivo muy grande a un negativo 
muy grande. 

4. Sustracción de un negativo muy grande a un positivo 
muy grande. 


Vamos a tratar de mejorar la anterior definición de desbor- 
damiento. 

Desde un punto de vista técnico, se reserva un bit especial, o 
indicador de desbordamiento, llamado “bandera” para cuando 
se produzca arrastre del bit 6 al bit 7, sin que haya acarreo 
externo, o cuando no haya arrastre del bit 6 al bit 7, pero si 
acarreo externo. Esto significa que el bit 7, y, por tanto, el 
signo del número, ha cambiado accidentalmente. Para el lector 
interesado por los aspectos técnicos, diremos que la bandera de 
desbordamiento se determina sometiendo a la operación O 
exclusiva los acarreos que llegan al bit 7 y los que salen de él. 
Prácticamente, todos los microprocesadores disponen de una 
bandera especial de desbordamiento que detecta automática- 
mente esa situación para que pueda emprenderse la acción 
correctora. 

La presencia de desbordamiento significa que el resultado de 
la suma o la resta exige más bits que los disponibles en el 
registro usual de ocho bits utilizado para almacenarlo. 


El acarreo y el desbordamiento 


Los bits de acarreo y desbordamiento se llaman “banderas”. 
Existen en todos los microprocesadores, y en el próximo capitu- 
lo aprenderemos a aprovecharlos para crear programas efecti- 
vos. Ámbos se encuentran en un registro especial llamado de 
banderas o de “estado”, que contiene, además, otros indicado- 
res, cuya función se abordará en el capítulo 4. 


Ejemplos 


Veamos a continuación el comportamiento del acarreo y el 
desbordamiento mediante algunos ejemplos prácticos. En todos 
los casos, el símbolo O denotará el desbordamiento, y el C, 
el acarreo. 


Si no hay desbordamiento O =0, y en caso contrario, 
O = 1; lo mismo para el acarreo C. Recuerde que las reglas de 
aplicación del complemento a dos exigen que se ignore el 
acarreo (aunque no daremos aquí demostración matemática de 
esta norma). 


Positivo-positivo 


00000110 (+6) 
+ 00001000 (+ 8) 


= 00001110 (+14 ою C:0 
(CORRECTO) 


Positivo-positivo con desbordamiento 


01111111 (+ 127) 
+ 00000001 (+1) 


= 10000000 (— 128) О: С:0 


El resultado es incorrecto, porque se ha producido des- 
bordamiento. 


(ERROR) 


Positivo-negativo (resultado positivo) 


00000100 (+ 4) 
+ 11111110 (— 2) 


=(1)00000010 (+2) Оо C:1 (se desprecia) 
(CORRECTO) 


Positivo-negativo (resultado negativo) 


00000010 (+ 2) 
+ 11111100 (- 4) 


= 11111110 (—2) O:0 C:0 
(CORRECTO) 


Negativo-negativo 


11111110 (—2) 
+ 11111110 (-4) 


= (1)11111010 (—6) O:0 С:1 (se desprecia) 


(CORRECTO) 


Negativo-negativo con desbordamiento 


10000001 (— 127) 
+ 11000010  (— 62) 


= (1101000011 (67) Ол C:1 
(ERROR) 


En este caso se ha producido desbordamiento por adición 
de dos números negativos muy grandes. El resultado es — 189 
y, por su magnitud, no cabe en ocho bits. 


Ejercicio 1.12: Resuélvanse las adiciones propuestas a continua- 
ción, indicando en cada caso el resultado, el acarreo C, el 
desbordamiento O y si el primero es o no correcto. 


10111111 () 11111010 (22-2 
+ 11000001 с) + 11111001 (___) 
= O C: о. _ Сс: _ 
О CORRECTO О CORRECTO 
[1 ERROR O ERROR 
00010000 (2) 01111110 C») 
+ 01000000 (_) + 00101010 С) 
= О: С: = O: С: 
O CORRECTO ПІ CORRECTO 
[] ERROR Г] ERROR 


Ejercicio 1.13: ¿Podría proponer un ejemplo de adición de un 
número positivo a otro negativo que causase un desbordamien- 
to? ¿Por qué? 


Representación en formato fijo 


Ya sabemos representar enteros con signo, pero todavía no 
hemos resuelto el problema de la magnitud. Para representar 
enteros grandes necesitamos varios bytes; pero para hacer 
operaciones artiméticas con eficacia hay que emplear un núme- 
ro de bytes fijo, no variable; por tanto, la determinación del 
número de bytes supone la del mayor número representable. 


Ejercicio 1.14: ¿Cuáles son los números máximo y mínimo repre- 
sentables con dos bytes en complemento a dos? 


El problema de la magnitud 


Al sumar números nos hemos limitado a trabajar con ocho 
bits, porque el microprocesador que vamos a utilizar opera 
internamente con grupos de ocho bits. Pero esto limita el 
campo de actuación a las cantidades comprendidas entre — 128 
у + 127, sin duda insuficientes para muchas aplicaciones. 

Para aumentar el número de cifras representables, se recurre 
a la precisión múltiple, que consiste en el empleo de formatos 
de dos, tres o n bytes. Veamos algunos ejemplos de un formato 
de doble precisión de 16 bits: 


00000000 00000000 es “0” 
00000000 00000001 es “1” 


01111111 11111111 es “32767” 
11111111 11111111 es *— 1” 
ИП ПИИЮ es “— 2” 


Ejercicio 1.15: ¿Cuál es el mayor entero negativo representable 
con un formato de triple precisión en complemento a dos? 


Pero el método tiene sus inconvenientes, Al sumar dos 
números, por ejemplo, habitualmente hay que hacerlo de ocho 
en ocho bits —la forma de operar se describirá en el capítu- 
lo 3, Técnicas elementales de programación—, lo que reduce la 
velocidad del proceso. Además, esta forma de representación 
adjudica 16 bits a todos los números, incluso a los que cabrían 
en ocho; en consecuencia, es normal utilizar 16 bits e incluso 
32, pero raramente más. 

Hay, además, otro punto importante que merece reflexión: 
sea cual sea el número и de bits elegido para la representación 
en complemento a dos, es fijo. Si el resultado, o un cálculo 
intermedio, genera un número cuya representación exige más de 
n bits, los que excedan se perderán; por lo general, el programa 
conservará los n de la izquierda (los más significativos) y 
rechazará los de orden inferior. Esta operación se llama truncar 
el resultado. 

Consideremos el siguiente ejemplo con representación de seis 
cifras en el sistema decimal: 


123456 
x 1.2 


246912 
123456 
= 148147.2 


Figura 1.4 
Tabla BCD. 
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El resultado necesita de siete cifras, y en el formato en que 
se trabaja el “2” situado tras el punto decimal se perdería, de 
manera que el resultado quedaría en 148 147. Por lo general, en 
la medida en que no se pierde la posición de la coma decimal, 
este método se utiliza para ampliar la cantidad de operaciones 
realizables a costa de la precisión. 

En el sistema binario el problema es el mismo. Los detalles 
de la multiplicación binaria se expondrán en el capítulo 4. 

La representación en formato fijo, aun con el riesgo de 
pérdida de precisión que ocasiona, puede bastar para realizar 
operaciones matemáticas normales. 

Por desgracia, la contabilidad no tolera ninguna inexactitud. 
Cuando se marca el total en una caja registradora, lo que debe 
aparecer es el precio exacto, no un valor aproximado; por 
tanto, cuando la precisión es incuestionable, hay que recurrir a 
otro tipo de representación, que, por lo general, es la llamada 
BCD, decimal codificado en binario. 


Representación BCD 


Esta técnica consiste en codificar cada una de las cifras 
decimales y utilizar todos los bits que sean necesarios para 
representar con exactitud el número completo. Para codificar 
las diez cifras comprendidas entre 0 y 9 hacen falta cuatro bits. 
Tres dan lugar a sólo ocho combinaciones, insuficientes para 
codificar diez cifras; con cuatro pueden hacerse dieciséis combi- 
naciones, que sí bastan para codificar las diez cifras decimales. 
Lo malo es que sobran seis posibles códigos (véase figura 1.4), 
que pueden causar problemas al sumar y restar. 


Се ен E 
1000 8 


1001 9 

1010 no utilizado 
no utilizado 
no utilizado 
no utilizado 
no utilizado 
no utilizado 


IA о м = © 


Como sólo hacen falta cuatro bits para codificar una cifra 
en BCD, pueden representarse dos cifras en cada byte, formato 
que se llama BCD condensado. 


= „== --------- снн ааа 


Así. "00000000" es “007 еп BCD y “10011001” es “99”. 
Un código BCD se lee como sigue: 


0010 0001 


Cifra BCD esl 
Cifra BCD "1" 
Número BCD "21" 


Ejercicio 1.16: ¿Cómo se representa "29" en BCD? ¿Cómo “91”? 


Ejercicio 1.17: ¿Es "10100000" una representación correcta en 
BCD? ¿Por qué? 


Para representar todas las cifras. se utilizan tantos bytes 
como sean necesarios. Por lo general, al principio de la repre- 
sentación se reservan uno o más nibbles para indicar el número 
total de nibbles. y, por tanto, de cifras BCD. También suele 
reservarse un nibble o un byte para indicar la posición de la 
coma decimal, aunque las convenciones adoptadas son varia- 
bles. 

He aquí un ejemplo de representación multibyte BCD de un 
entero: 


ТӘРЕП ӨСЕ КЕЗЕ UNE ON ШЕ. | | (3 bytes) 


r | número “221” 


número 
de cifras 
(hasta 255)signo 


El número representado es + 221 (en cuanto al signo, 0000 
puede representar +, por ejemplo, y 0001, —). 


Ejercicio 1.18: Utilizando la misma convención en cuanto al signo, 
represéntese el número “— 23 123” en formato BCD, como en 
el ejemplo, y en notación binaria. 


Ejercicio 1.19: Represéntese en BCD “222”, "111" y el resultado 
de la operación 222 x 111. ( Ejecütese la operación a mano.) 


El sistema BCD se adapta muy fácilmente a la representa- 
ción de nümeros decimales. 
Así, + 2.2] podría representarse como sigue: 


1? cifra 24 cifra 32 cifra 


3 cifras el ^." está a la + 
izquierda de la 
2? cifra 


La ventaja del sistema BCD es que arroja resultados riguro- 
samente exactos, aunque a costa de ocupar mucha memoria y 
reducir la velocidad de ejecución de las operaciones aritméticas. 
Este inconveniente sólo se acepta en el campo de la contabili- 
dad, y habitualmente el sistema no se emplea en otros casos. 


Ejercicio 1.20: ¿Cuántos bits hacen falta para codificar “9999” en 
BCD? ¿Y en complemento a dos? 


Ya hemos resuelto los problemas asociados a la representa- 
ción de enteros, de enteros con signo y hasta de grandes 
enteros. También hemos visto un procedimiento para represen- 
tar decimales en BCD. Veamos a continuación el problema que 
supone la representación de nümeros decimales en un formato 
de longitud fija. 


Representación en punto flotante 


La condición básica es que los decimales deben representar- 
se en un formato fijo. Para no desperdiciar bits, la representa- 
ción utilizada pasará por la normalización de todos los nümeros. 

Al escribir "0.000123" se desperdician tres ceros a la izquier- 
da del nümero, ceros que sólo sirven para indicar la posición 
del punto. El nümero podría normalizarse escribiéndolo .123 
x 1073; “123” se llama mantisa normalizada, y “— 3”, exponen- 
te. La normalización ha consistido en la eliminación de todos 
los ceros situados a la izquierda y en el cálculo del exponente. 

Veamos otro ejemplo: 22.1 se normaliza como .221 x 102, y, 
en general, cualquier número será igual а M x 10E, siendo M la 
mantisa y E el exponente. 

Como se comprueba inmediatamente, la mantisa de un 
número normalizado es menor que | y mayor o igual que 0.1, 
siempre que aquél no sea nulo. Es decir: 


1<M<! ó 10 !<М «10? 
De la misma manera, en representación binaria: 
21 1<M<2 (6 5<М<1) 


Figura 1.5 


Representación en punto flo- 


tante típica. 


siendo M el valor absoluto de la mantisa con independencia del 
signo). 
Por ejemplo: 


111.01 se normaliza como .11101 x 2? 


siendo la mantisa 11101 y 3 el exponente. 

Ahora que ya hemos expuesto el fundamento de la represen- 
tación, pasemos a examinar el formato real, que aparece en la 
figura 1.5. 


31 24 23 16 15 8 7 0 


En la representación utilizada en este ejemplo se emplean un 
total de cuatro bytes con 32 bits. El primer byte por la 
izquierda contiene el exponente, que, al igual que la mantisa. se 
representa en complemento a dos; esto significa que el máximo 
exponente posible es — 128. En la figura 1.5, “S” denota el bit 
de signo. 

Para representar la mantisa se utilizan tres bytes; dado que 
el primer bit en complemento a dos corresponde al signo, el 
formato deja 23 bits para albergar la magnitud de la mantisa. 


Ejercicio 1.21: ¿Cuántas cifras decimales pueden representarse con 
una mantisa de 23 bits? 


Esto no es más que un ejemplo de representación en punto 
flotante. En la práctica pueden usarse sólo tres bytes o más de 
cuatro. La de cuatro bytes propuesta es una bastante usual, que 
constituye un compromiso razonable entre exactitud, magnitud 
representable, aprovechamiento de la memoria y eficacia opera- 
tiva. 

Ya hemos explorado los problemas asociados a la represen- 
tación de números enteros y decimales, con y sin signo. Pode- 
mos, pues, pasar a ocuparnos de la representación interna de 
datos alfanuméricos. 


Representación de datos alfanuméricos 


La representación de datos alfanuméricos, es decir, de carac- 
teres, es sencillísima: los caracteres se traducen a un código de 
ocho bits. En informática no hay más que dos códigos de uso 
general, el АЗСП y el EBCDIC. ASCII son las siglas en inglés 
de Código Normalizado Americano para el Intercambio de 
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БІЗ 2-27-20 :25-2 2722720 0- 


Información, y es de uso universal еп microprocesadores. El 
código EBCDIC es una variante del ASCII utilizada por IBM 
que, por tanto, sólo la utilizan los microordenadores en las . 
conexiones con terminales IBM. 

Examinemos brevemente el código ASCII. Se trata de codifi- 
car 26 letras del alfabeto en minúsculas y mayúsculas, 10 
símbolos numéricos y alrededor de 20 símbolos especiales 
(a efectos de codificación, la М se considera un signo especial, no 
usado en inglés). Todo ello puede hacerse fácilmente con 7 bits, 
que proporcionan 128 combinaciones diferentes (véase figura 
1.6). Sin embargo, antes hemos hablado de 8 bits. ¿Para qué 
sirve el octavo? Este octavo bit es el bit de paridad, y sirve para 
garantizar la conservación del contenido de un byte. Funciona 
de la siguiente manera: se cuenta el total de unos de los siete 
primeros bits y, si es impar, el octavo se iguala a 1, para que 
pase a ser par. Es lo que se llama paridad par (también puede 
trabajarse con paridad impar, que consiste en calcular el octavo 
bit —el de la izquierda— de manera que el total de unos sea 
impar. 

Ejemplo: Calcúlese el bit de paridad de “0010011” en pari- 
dad par. El número de unos es tres, de manera que el bit de 
paridad ha de ser 1 para que pase a ser cuatro; por tanto, par: 
10010011; el primer 1 es el bit de paridad, y 0010011 es el 
carácter. 

La tabla de códigos ASCII de 7 bits aparece en la figura 1.6. 
En la práctica se utiliza “tal cual”, es decir, sin paridad, 
añadiendo un O en la posición izquierda, o con paridad, aña- 
diendo en esa misma posición el bit adecuado. 


3 4 5 6 7 
mn 011 100 101 110 111 
0 0 Q P - p 
1 1 А Q a а 
2 2 в в b r 
3 3 C S c s 
4 4 D T d t 
5 5 Е U е u 
6 6 F V f у 
7 т а W g w 
8 8 H х h x 
9 9 | Y i y 
A : J 2 j 2 
B бок E k { 
С < L \ 2 әде 
Figura 1.6 B M m 
Table ае саа Е > N ] j b 
veanse ias аогематиг 
dis B). F ? о «— o DE 
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Ejercicio 1.22: Calcúlese la representación en 8 bits de las cifras 
70” a "9" con paridad par (el código resultante se utiliza en 
los ejemplos de aplicaciones del capítulo 8). 


| Ejercicio 1.23: Idem de las letras "A" a “F”. 


Ejercicio 1.24: Indíquense los contenidos binarios de los cuatro 
caracteres propuestos a continuación, utilizando para ello el 
código ASCII sin paridad (el bit de la izquierda es, por tanto, 
“0”: 


“А” 
“9% 
«9 
“h” 


En situaciones especiales —las telecomunicaciones, por ejem- 
plo— se emplean códigos especiales de corrección y de otras 
clases, pero están fuera del alcance de este libro. 

Una vez estudiadas las formas de representación más usua- 
les en el interior del ordenador para el programa y para los 
datos, pasaremos a examinar las posibles representaciones ex- 
ternas. 


REPRESENTACION EXTERNA DE LA INFORMACION 


La representación externa es la forma en que la información 
se ofrece al usuario, que, por lo general, es el programador. 
Puede ser binaria, octal o hexadecimal y simbólica. 


1. Binaria 


Como hemos visto, la información se almacena internamen- 
te en bytes, que son secuencias de ocho bits (ceros o unos). A 
veces es deseable presentar esta información interna directamen- 
te en su formato binario; es lo que se llama representación 
binaria. Un ejemplo sencillo lo proporcionan los diodos lumino- 
sos, LED, del teclado de algunos microordenadores. Si el 
microprocesador es de ocho bits, en el teclado habrá probable- 
mente LED para exteriorizar el contenido de cualquiera de los 
registros (un registro, que se describirá en el capítulo 2, contiene 
ocho bits de información): un piloto iluminado denota un 1, y 
otro apagado, un 0. La representación binaria es útil para 
corregir con detalle programas complejos, sobre todo si inclu- 
yen operaciones de entrada y salida, pero constituye una forma 
de comunicación obviamente poco práctica. Por eso se prefiere 
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casi siempre la representación simbólica (en efecto, es mucho 
más fácil entender y recordar “9” que “1001”). Se han ideado 
formas de comunicación más cómodas que mejoran la relación 
hombre-máquina. 


2. Octal y hexadecimal 


En los sistemas octal y hexadecimal se codifican tres y Я 
cuatro bits binarios, respectivamente, еп un único simbolo. En 
el primero de ellos, cualquier combinación de tres bits binarios 
se representa mediante un número comprendido entre 0 y 7; la 
figura 1.7 recoge la tabla de símbolos de este sistema. 


Binario 
|. 


000 
001 
010 
011 
100 
101 
110 


Figura 1.7 
Símbolos octales. 


Así, el número binario “00 100 100” se representa en octal 
como “044”. 
0 4 4 


Otro caso: 11 111 111 es “377” en octal. 
Y Y Y 
3 7 1] 


E inversamente, el número octal “211” equivale a 
010 001 001 


es decir, “10001001” en binario. 

El sistema octal se utilizaba tradicionalmente en los ordenado- 
res antiguos, que trabajaban con un número de bits comprendi- : 
do habitualmente entre 8 y 64. Actualmente, cuando el dominio 
de los microprocesadores de 8 bits ha convertido a este formato 
en la norma, resulta más práctico recurrir a la representación 
hexädecimal. 

En el sistema hexadecimal, cada grupo de cuatro bits se 
codifica como una cifra hexadecimal. Estas se representan 
mediante los símbolos 0 a 9 más las letras A, B, C, D, E y F. 
Así, "0000" es “0”, “0001” es “1” y “1111” es “F” (véase la figura 1.8). 


“ 


Figura 1.8 
Códigos hexadecimales. 


0 0 0 


0000 


0001 1 


0010 


0011 


0100 


0101 


0110 


0111 


1000 


1001 


1010 


1011 


1100 
1101 


1110 


1111 


Ejemplo: el número binario 1010 0001 equivale 
—— ~ а 


al hexadecimal А 1 


Ejercicio 1.25: ¿Cuál es la representación hexadecimal de 
“10101010? 


Ejercicio 1.26: ¿Cuál es el equivalente binario del hexadecimal 
“ЕА”? i 


Ejercicio 1.27: ;Cuál es la representación octal de "01000001"? 


El sistema hexadecimal tiene la ventaja de que codifica ocho 
bits en sólo dos cifras, lo que es más fácil de visualizar y 
memorizar y más rápido de teclear en el ordenador que el 
equivalente binario. Por ello, en la mayor parte de los nuevos 
microordenadores se prefieren representar los grupos de bits en 
el sistema hexadecimal. 
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Naturalmente, cuando la información presente en la memo- 
ria tenga significado —un texto o una serie de números— el 
sistema hexadecimal no constituirá un método de representa- 
ción adecuado para el hombre. 


Representación simbólica 


Se llama así a la representación externa de la información en 
su forma simbólica real. Los números decimales, por ejemplo, se 
representan como tales, y no como secuencias de símbolos 
hexadecimales o de bits; de la misma manera, el texto escrito 
adopta la forma de una sucesión de letras. Para el usuario, esta 
forma de representación es obviamente la más práctica, y se 
utiliza siempre que se cuente con un dispositivo de visualización 
adecuado, como un monitor de televisión o una impresora. Por 
desgracia, tales dispositivos siguen resultando excesivamente 
costosos para los microordenadores más elementales, que, por 
tanto, se comunican con el usuario en el sistema hexadecimal. 


RESUMEN DE REPRESENTACIONES EXTERNAS 


La representación simbólica es la más deseable, ya que es la 
más natural para el hombre. Sin embargo, exige un interfaz 
costoso en forma de teclado alfanumérico e impresora o, moni- 
tor TV, lo que la hace incompatible con los sistemas más 
baratos. En esas situaciones, la alternativa más frecuente es el 
sistema hexadecimal. La representación binaria se emplea única- 
mente en la puesta a punto muy detallada de los soportes 
lógico o físico; en esta forma de representación se visualiza 
directamente el contenido de los registros internos de la memo- 
ria. (La utilidad de la visualización binaria directa en el panel 
de mando del ordenador ha sido siempre motivo de acaloradas 
discusiones, en las que no entraremos aquí.) 

Estudiadas las diversas técnicas de representación interna y 
externa de la información, estamos en condiciones de pasar a 
examinar el microprocesador que se encargará de manipularla. 


EJERCICIOS ADICIONALES 


Ejercicio 1.28: ¿Qué ventajas tiene la representación en comple- 
mento a dos sobre otras en el manejo de números con signo? 


Ejercicio 1.29: ¿Cómo se representaría “1024” en las notaciones 
binaria directa, binaria con signo y complemento a dos? 


Ejercicio 1.30: ¿A qué se llama bit 0? ¿Debe el programador 
comprobarlo tras una adición o una sustracción? 


Ejercicio 1.31: Calcúlense los complementos a dos de “+ 16”, 
113 Cas 1 7”, ес + 18”, 113 EN 16", [14 E 17" y “ до 4 18”. 


Ejercicio 1.32: Indiquese la representación hexadecimal del texto 


“MENSAJE”, almacenado internamente en formato ASCII 
sin paridad. 
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Lud 
am. 
E: 
ще 
в 
se 
в 
a 
me 
E 
ж 
E 


ui 

же 
ee 
же. 
ав 
ES 
Яғ 
же 
ж 
ж 
EJ 
m 


Organización del 


Introducción 


ardware 
del 780 


Para hacer programas sencillos no es necesario conocer en 
detalle la estructura interna del procesador que se está utilizan- 
do, pero sí es imprescindible tal conocimiento si se pretenden 
crear programas más ambiciosos. La finalidad de este capitulo 
es presentar los aspectos básicos de la constitución física del 
sistema Z80 indispensables para entender su funcionamiento. 
Un microordenador completo consta de varios dispositivos, 
además del microprocesador (el Z80 en este caso); nos limitare- 
mos еп esta parte del libro a examinar el Z80 propiamente . 
dicho, y dejaremos el resto de los dispositivos (en su mayor 
parte componentes de entrada y salida) para más adelante (ca- 
pítulo 7). 

Repasaremos, primero, la arquitectura básica del microorde- 
nador; a continuación examinaremos con más detalle la organi- 
zación interna del Z80 y, en particular, sus diversos registros, 
A ello seguirá el estudio de la ejecución de programas y del 
mecanismo de secuenciación. Pero este capítulo dará una visión 
un tanto simplificada del hardware; el lector verdaderamente 
interesado en ello deberá consultar el libro Microprocessors, del 
mismo autor. 
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El Z80 se diseñó para sustituir al Intel 8080 y ampliar, de 
paso, sus posibilidades; a lo largo del texto nos referiremos en 
varias ocasiones al 8080 de Intel. 


Arquitectura del sistema 


Figura 2.1 
Sistema Z80 normal. 


La arquitectura del microordenador se ilustra en la fi- 
gura 2.1. El microprocesador (иР), el 780 en este caso, está 
situado en la parte izquierda del esquema, y desempeña las 
funciones de una unidad central de proceso (CPU), contenida 
integramente en un solo microcircuito monolítico de silicio; 
consta de una unidad aritmética y lógica (ALU), con sus propios 
registros internos, y una unidad de control (UC), encargada de la 
secuenciación de las operaciones que ha de realizar el sistema. 
Veremos su funcionamiento dentro de este mismo capítulo. 


osc Loly 
PUERTO A 
+5 
280 


BUS DE DIRECCIONES 


Вав. 


+5V MASA 


El ИР dispone de tres buses o canales conductores de 
información: un bus de datos bidireccional de 8 bits, representa- 
do en la parte superior de la ilustración; un bus de direcciones 
unidireccional de 16 bits, y un bus de control, esquematizado 
junto con el anterior en la parte inferior de la figura. Veamos 
ahora la función que cumple cada uno de ellos. 

El bus de datos transporta datos de unos elementos del 
sistema a otros, típicamente de la memoria al иР y viceversa, о 
del иР a circuitos de entrada y salida (estos circuitos de entrada y 
salida son los componentes encargados de poner el sistema еп 
comunicación con dispositivos externos). 

El bus de direcciones lleva las direcciones generadas por el 
иР, que seleccionan entre los registros internos dentro de los 
microcircuitos conectados al sistema. Dicha dirección especifica 
la fuente —o el destino— del dato que debe circular a lo largo 
del bus de datos. 


El bus de control conduce las diversas señales de sincroniza- 
ción que gobiernan el funcionamiento del sistema. 

Una vez descrito el papel que desempeñan los buses, pasare- 
mos a conectar al sistema los demás componentes necesarios 
para su funcionamiento. 

El uP necesita una referencia de tiempo exacta, que propor- 
cionan un reloj y un cristal. En los microprocesadores más 
“antiguos”, el reloj-oscilador es externo y, por lo general, adopta 
la forma de otro circuito integrado. En los modelos más 
recientes, el mencionado reloj suele ir incorporado en el propio 
иР. Sin embargo, el cristal de cuarzo, debido a su tamaño, es 
siempre exterior al sistema. En la figura 2.1, reloj y cristal se han 
esquematizado a la izquierda del recuadro correspondiente al иР. 

Veamos ahora el resto de los elementos del sistema. Avan- 
zando de izquierda a derecha en la figura encontramos: la 
memoria sólo de lectura о memoria fija (read-only memory, 
ROM), que contiene el programa del sistema. La ventaja de la 
ROM es que su contenido es permanente; no desaparece ni 
siquiera cuando se desconecta el ordenador. Por ello se utiliza 
siempre para almacenar un programa de control (cuyo funciona- 
miento explicaremos más adelante) encargado de garantizar la 
puesta en marcha del sistema. En aplicaciones de control de 
procesos, casi todos los programas deben almacenarse en ROM, 
porque normalmente no sufrirán modificaciones y porque, ade- 
más, deben protegerse frente a cortes de corriente accidentales. 

Por el contrario, los aficionados y los profesionales dedica- 
dos a la creación, desarrollo y comprobación de programas 
almacenan éstos en memoria КАМ, para poder modificarlos 
con facilidad. Más adelante, esos programas pueden transferirse 
a una memoria ROM o dejarse en RAM, aunque el contenido 
de esta memoria se volatiliza cuando se interrumpe la corriente. 

La memoria RAM (random-access memory, memoria de acce- 
so aleatorio) es la memoria de lectura y escritura del sistema. 
En un sistema de control, la capacidad de RAM suele ser 
reducida, ya que habitualmente se utiliza sólo para almacenar 
datos; por el contrario, los sistemas destinados al desarrollo de 
programas necesitan una memoria RAM de gran capacidad, 
con espacio suficiente para albergar programas y material lógi- 
co en desarrollo. El contenido de la memoria RAM debe car- 
garse de un dispositivo externo antes de usarla. 

Por último, el ordenador necesita uno o más circuitos inte- 
grados de conexión para comunicarse con el mundo exterior. El 
circuito de conexión más usado es el llamado PIO (parallel 
input/output, entrada y salida en paralelo), que se muestra en la 
figura. Este PIO, como todos los demás circuitos del siste- 
ma, está conectado a los tres buses y proporciona, al menos, 
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dos puertos de 8 bits para facilitar la comunicación con el 
mundo exterior. Para más detalles sobre el funcionamiento real 
del PIO, consulte el libro Microprocessors; si le interesa su 
aplicación específica al sistema Z80, más adelante, en el capítu- 
lo 7 de este libro (Dispositivos de entrada y salida), se trata. 

Todos los microcircuitos están conectados a los tres buses, 
incluido en todos los casos el de control. 

Pero no todos los módulos funcionales que acabamos de 
describir tienen que estar integrados en una misma pastilla LSI 
(large scale integration, de integración a gran escala). Lo normal 
es, por el contrario, combinar varios circuitos, como un PIO, y 
varias ROM o RAM. 

Acabamos de describir los componentes esenciales, pero pa- 
ra que el sistema funcione hacen falta algunos más: así, los 
buses deben estar, por lo general, protegidos por amplificadores- 
separadores; los microcircuitos RAM necesitan un circuito lógi- 
co de decodificación; por último, algunas señales deben ser, 
asimismo, intensificadas por separadores. Aquí no describiremos 
estos circuitos auxiliares, puesto que no influyen en la progra- 
mación. El lector interesado en las técnicas de montaje y crea- 
ción de conexiones deberá consultar el volumen, Microprocessor 
Interfacing Techniques. 


El interior del microprocesador 


La inmensa mayoría de los circuitos integrados microproce- 
sadores que actualmente existen en el mercado tienen la misma 
arquitectura que describiremos aquí y que muestra la figura 2.2. 
Empezaremos a examinar con detalle los módulos que compo- 
nen el microprocesador a partir del extremo derecho de la 
ilustración. 

La casilla de control situada en la parte derecha representa 
la unidad de control, encargada de sincronizar la actividad de 
todo el sistema. Su comportamiento se irá aclarando a lo largo 
de lo que queda de capitulo. 

La ALU realiza operaciones aritméticas y lógicas. Una de las 
entradas de la ALU, la situada a la izquierda en este caso, está 
equipada con un registro especial llamado acumulador (puede 
haber varios acumuladores). Dentro de la misma instrucción, el 
acumulador puede referenciarse como entrada y como salida 
(fuente y destino). 

La ALU se encarga también de las operaciones de desplaza- 
miento y rotación de bits. 

Se llama desplazamiento a la traslación del contenido de un 
byte una o más posiciones hacia la izquierda o hacia la derecha 


Figura 2.2 
Estructura habitual de un mi- 
croprocesador. 


Figura 2.3 
Desplazamiento y rotación. 


BUS EXTERNO DE DATOS 


BUS INTERNO (8 BITS) 


ACUMULADORE 


= ODV- От 


REGISTROS DE 
DATOS DE 
8 BITS 


BUS EXTERNO 
DE DIRECCIONES 
(16 BITS) 


(véase la figura 2.3). Cada uno de los bits avanza una posición 
hacia la izquierda. Los detalles de estas dos instrucciones se 
estudiarán en el próximo capítulo. 


DESPLAZAMIENTO А LA IZQUIERDA 


ACARREO 


ROTACION A LA IZQUIERDA 


ACARREO 


Nota: Algunas instrucciones de desplazamiento y rotación по incluyen acarreo. 


El desplazador puede estar situado en la salida de la ALU, 
como en la figura 2.2, o en la entrada del acumulador. 

A la izquierda de la ALU están las banderas о registro de 
estado. Su función es “llevar la cuenta” de las situaciones excep- 
cionales que se dan en el interior del microprocesador. El conte- 
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nido del registro de estado puede verificarse mediante instrúc- 
ciones especiales o leerse en el bus interno de datos. Las ins- 
trucciones condicionales provocan la ejecución de un nuevo 
programa en función del valor de uno de esos bits. 

La función de los bits de estado en el Z80 se estudiará más 
adelante en este mismo capitulo. 


ACTIVACION DE LAS BANDERAS 


La mayor parte de las instrucciones ejecutadas por el proce- | 
sador modificarán alguna de las banderas o todas ellas. Es 
importante consultar siempre la tabla del fabricante, que indica 
qué bits serán modificados por las instrucciones, ya que ésa es 
la única forma de comprender cómo se va desarrollando el 
programa. La figura 4.17 recoge una tabla como la mencionada 
para el Z80. 


LOS REGISTROS | 


Volvamos a la figura 2.2. A la izquierda del esquema se 
observan los registros del microprocesador, dividios conceptual- 
mente en dos categorías: registros de tipo general y registros de 
direcciones. 


REGISTROS DE TIPO GENERAL 


Los registros de tipo general deben organizarse en orden 
para que la ALU manipule los datos a velocidad elevada. 
Debido a las limitaciones del número de bits que resulta razo- 
nable proporcionar dentro de una instrucción, casi siempre hay 
menos de ocho registros (directamente direccionables) Cada 
uno de ellos es un conjunto de ocho elementos biestables conec- 
tados al bus bidireccional interno de datos. Los ocho bits pue- 
den entregarse simultáneamente al mencionado bus o recibirse 
desde el mismo. Los registros biestables en MOS (tipo de dispo- 
sitivo, metal-óxido-semiconductor) constituyen la forma de me- 
moria más rápida existente, y el acceso a su contenido se lleva a 
cabo en algunas decenas de nanosegundos. 

Los registros internos están habitualmente etiquetados de 0 а 
n, y su función no está definida de antemano (por eso se dice 
que son de tipo general). Pueden contener cualquier dato utili- 
zado por el programa. 


Estos registros de tipo general suelen utilizarse para almace- 
nar datos de ocho bits. En algunos microprocesadores pueden 
manipularse dos de esos registros simultáneamente, que se lla- 
man “registros apareados” y que permiten el almacenamiento 
de magnitudes —datos o direcciones— de 16 bits. 


REGISTROS DE DIRECCIONES 


Son registros de 16 bits destinados especificamente al alma- 
cenamiento de direcciones, que con frecuencia se conocen tam- 
bién como contadores de datos о apuntadores. Su caracteristica 
principal es que están conectados al bus de direcciones. De 
hecho, los registros de direcciones crean el bus de direcciones 
(éste aparece en la parte inferior izquierda de la figura 2.4). 

Estos registros de 16 bits sólo pueden cargarse por medio 
del bus de datos, lo que obliga a realizar dos transferencias de 8 
bits. Para diferenciar las mitades inferior y superior de cada 
registro, suele llamarse a la primera L (inferior, lower, que 
comprende los bits О a 7), y H (superior, higher, que comprende 
los bits 8 a 15), a la segunda; tales etiquetas se emplean siempre 
que es necesario distinguir entre las dos mitades. En la mayor 
parte de los microprocesadores hay, al menos, dos registros de 
direcciones (véase la figura 2.4; “MUX” es abreviatura de multi- 
plexor). 


BUS DE DATOS (8 BITS) 


REGISTROS DE 
DIRECCIONES 
DE 16 BITS 


Figura 2.4 

Los registros de direcciones de 
16 bits crean el bus de direccio- 
nes. 


BUS DE DIRECCIONES (16) 


Contador del programa (PC) 


Es un elemento imprescindible en cualquier procesador. 
Contiene la dirección de la instrucción que ha de ejecutarse a 
continuación. El mecanismo de ejecución del programa y la 
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secuenciación automática que se lleva acabo con ayuda del 
contador del programa se describirán en la siguiente sección. 
Por el momento, diremos que la ejecución de un programa es 
normalmente secuencial; para acceder a la instrucción siguiente 
es preciso extraerla primero de la memoria interna del micro- 
procesador. Para ello, se entrega el contenido del PC al bus de 
direcciones, que lo transmite a la memoria; ésta lee el contenido 
especificado por la dirección y devuelve la palabra correspon- 
diente, que es la instrucción, а la uP. En unos pocos micro- 
procesadores, como el F8 de dos microcircuitos integrados, no 
hay PC. Esto no significa que el sistema no tenga ese elemento, 
sino que, por razones de eficacia, actúa directamente en el 
circuito integrado de memoria. 


Puntero de la pila SP 


La pila se describirá en la próxima sección. En los micropro- 
cesadores de tipo general más potentes, la pila suele realizarse 
en la memoria mediante el soporte lógico. Para identificar el 
elemento superior de la pila dentro de aquélla se reserva un 
registro de 16 bits llamado puntero de la pila o SP (stack 
pointer ). El SP contiene la dirección del elemento superior de la 
pila dentro de la memoria. Como se verá, la pila es indispensable 
para programar interrupciones y acceder a las subrutinas. 


Registro de índice (IX) 


La adjudicación de índices es una operación de direcciona- 
miento que no está presente en todos los microprocesadores. En 
el capítulo 5 se describirán las diversas técnicas de direcciona- 
miento. El empleo de índices permite acceder con una sola 
instrucción a bloques completos de datos contenidos en la 
memoria. El registro de índice suele contener un valor de des- 
plazamiento que se suma automáticamente a una base (o vice- 
versa, una base que se añade a un desplazamiento); de esta 
forma, el índice da acceso a cualquiera de las palabras de un 
bloque de datos. 


LA PILA 


Una pila es lo que formalmente se llama una estructura 
LIFO (last-in first-out, último en entrar, primero en salir). Está 
formada por un conjunto de registros, o posiciones de memoria, 
adscritos a dicha estructura de datos. Su característica esencial 
es que se trata de una estructura cronológica; el primero de los 
elementos introducidos en la pila ocupa siempre el fondo de la 
misma, mientras que la introducción más reciente está siempre 
en su parte superior. Su organización es comparable a la del 


portabandejas de la barra de una hamburguesería; las bandejas 
| se apilan sobre una base que se hunde en el pozo de la barra 
с conforme aumenta el peso; se colocan y se cogen por arriba, de 
' manera que las últimas en llegar al montón son siempre las 

primeras en salir de él. Este ejemplo ilustra también otra pecu- 

liaridad de la pila, a saber: que sólo es accesible, mediante dos 
| instrucciones: empujar y extraer (PUSH y POP). 
| La operación de empujar consiste en la carga de un elemento 
(dos en el caso del Z80) en lo alto de la pila. Por extracción se 
entiende la toma de un elemento de la pila. En un microproce- 
sador es el acumulador lo que se coloca en la parte superior de 
| la pila, de manera que la extracción consiste en la transferencia 
a éste del último elemento de aquélla. Puede haber también 
otras instrucciones para llevar el elemento superior de la pila a 
otros registros particulares, como el de estados, por ejemplo; a 
este respecto, el Z80 es más flexible que casi todos los demás 
р microprocesadores. 

La pila es necesaria para aplicar tres recursos de programa- 
ción: subrutinas, interrupciones y almacenamiento temporal de 
datos. El papel que desempeña la pila en la ejecución de subru- 
tinas se explicará en el capítulo 3 (Técnicas básicas de progra- 
mación). Su función en las interrupciones la veremos en el 
capítulo 6 (Técnicas de entrada y salida). Por último, el com- 
portamiento de la pila como almacenamiento temporal de datos 
en el proceso a alta velocidad se expondrá durante los progra- 
, mas de aplicaciones específicas. 

Por el momento, nos contentarmeos con aceptar que la pila 
es un componente imprescindible en cualquier sistema informá- 
tico. Se crea de dos formas: 


| 1. La primera consiste en reservar un número fijo de regis- 
tros dentro del propio microprocesador, es lo que se 
llama una pila en soporte físico. Tiene la ventaja de la 
velocidad y el inconveniente del número limitado de 
registros. 


2. En casi todos los microprocesadores de tipo general la 
| pila se realiza en el soporte lógico para no limitarla a un 
nümero muy reducido de registros. Es el procedimiento 
empledo en el Z80. Dentro del microprocesador se reser- 
va un registro, el SP en este caso, para almacenar el 
puntero de la pila, es decir, la dirección de su elemento 
superior (o, a veces, la dirección del elemento superior 
más uno). À continuación se crea la pila como una zona 
de la memoria; de esta forma bastan los 16 bits que 
| ocupa el puntero para señalar cualquier posición de 
aquélla. | 
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Figura 2.5 


Las dos instrucciones de mani- 
pulación de la pila. 


Figura 2.6 
Tomar una 
memoria. 
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CICLO DE EJECUCION DE LAS INSTRUCCIONES 


Examinemos la figura 2.6. La unidad microprocesadora apa- 
rece a la izquierda, y la memoria, a la derecha. Esta puede estar 
formada por microcircuitos ROM o RAM o de cualquier otra 
clase con capacidad de almacenamiento de datos e instruccio- 
nes. Veremos en este caso cómo se toma una instrucción de la 
memoria para ilustrar la función del contador del programa; 
supondremos que el contenido de éste es válido; ahora tiene 
una dirección de 16 bits, que corresponde a la siguiente instruc- 
ción, la que debe tomarse de la memoria. Todos los procesado- 
res siguen un ciclo de tres partes: 


1. Tomar la siguiente instrucción. 
2. Decodificar la instrucción. 
3. Ejecutar la instrucción. 


ИР ВОМ/ВАМ 


2222 INSTRUCCION; 


BUS DE DIRECCIONES 
РРР, 


Tomar 


En la primera parte del ciclo, el contenido del contador del 
programa se deja en el bus de direcciones, que lo conduce a la 
memoria. Simultáneamente, caso de que sea necesario, se emite 
una señal de lectura a lo largo del bus de control del sistema. 


Figura 2.7 


Secuenciación automática. 


La memoria recibe la dirección, que especifica una de sus posi- 
ciones. Al recibir la señal de lectura, decodifica la dirección por 
medio de su propio decodificador y selecciona la posición indi- 
cada en ella. Algunos cientos de nanosegundos más tarde, la 
memoria deja el dato de ocho bits correspondiente a la direc- 
ción pedida en el bus de datos. Esta palabra de ocho bits es la 
instrucción que se quería tomar. En la figura, la instrucción se 
entrega al bus de datos esquematizado en la parte superior 
del uP. 

Resumamos brevemente la secuencia: el contenido del conta- 
dor del programa pasa al bus de direcciones; se genera una 
señal de lectura; la memoria realiza su ciclo y, unos trescientos 
nanosegundos más tarde, la instrucción situada en la dirección 
pedida pasa al bus de datos (suponiendo que la instrucción 
ocupe un solo byte). El microprocesador lee el bus de datos y 
deposita su contenido en un registro interno especial llamado 
IR o registro de instrucción; es un registro de ocho bits que se 
utiliza para almacenar la instrucción que se acaba de tomar de 
la memoria. El ciclo tomar ya se ha cerrado. Los ocho bits de 
la instrucción ya están físicamente alojados en el registro inter- 
no especial, IR, del иР. Este registro, situado a la izquierda 
en la figura 2.7, no es accesible al programador. 


Ж LL LL LV %% LL7) 


BUS DE DATOS ` 


MEMORIA 
0 


E ер. 


SEÑALES LECTURA 


ESPACIO PROPIO 
DE LA MEMORIA 


Decodificación y ejecución 

Una vez alojada la instrucción en el IR, la unidad de control 
decodifica su contenido y genera la secuencia de señales internas 
y externas necesarias para ejecutar la instrucción especificada. 
Hay, pues, un breve retardo de decodificación al que sigue una 
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fase de ejecución, cuya duración depende de la naturaleza de la 
instrucción. Algunas se ejecutan integramente dentro del 
ИР, mientras que otras toman datos de la memoria o los 
dejan en ella. Por eso, cada tipo de instrucción necesita un 
tiempo de ejecución diferente. Este tiempo se mide en ciclos de 
reloj (en el capítulo 4 se detallan los ciclos empleados por cada 
una: de las instrucciones). Los tiempos se expresan en ciclos, y 
no en nanosegundos, porque la duración del ciclo es variable. 


TOMAR LA SIGUIENTE INSTRUCCION 


Ya hemos descrito cómo, con ayuda del contador del pro- 
grama, se toma una instrucción de la memoria. Durante la 
ejecución de un programa, las instrucciones se toman de la 
memoria en secuencia, y para ello es preciso prever un mecanis- 
mo automático, que adopta la forma de un sencillo sumador 
conectado al contador del programa (figura 2.7). Su funciona- 
miento es el siguiente: cada vez que se coloca en el bus de 
direcciones el contenido del contador del programa (parte infe- 
rior de la ilustración), dicho contenido se incrementa en una 
unidad y se vuelve a escribir en el contador. St, por ejemplo, el 
contador tiene el valor “0”, éste pasa al bus de direcciones, 
mientras aquél pasa a valer “1”. De esta forma, cuando vuelva a 
funcionar traerá la instrucción de la dirección 1. Este ciclo cons- 
tituye un mecanismo automático de secuenciación de instrucciones. 

La descripción del párrafo anterior es, en realidad, una 
simplificación de la realidad, porque algunas instrucciones pue- 
den tener dos y hasta tres bytes de longitud, lo que obliga a 
tomarlos de la memoria uno tras otro; sin embargo, el mecanis- 
mo fundamental es idéntico: el contador del programa se usa 
tanto para tomar bytes sucesivos de una instrucción como para 
tomar sucesivas instrucciones. Por tanto, el dispositivo formado 
por el contador y el sumador señala automáticamente posicio- 
nes sucesivas de la memoria. 

Vamos ahora a ejecutar una instrucción dentro del иР 
(véase la figura 2.8). Una instrucción típica sería, por ejemplo, 
RO = RO + КІ, que significa: “SUMAR el contenido de RO al 
de R1 y almacenar el resultado en RO”. Para realizar la opera- 
ción, se lee el contenido del registro RO, que, a través del bus 
único, se lleva a la entrada izquierda de la ALU y se almacena 
en el registro auxiliar* situado en dicho punto. А continuación 
pasa al bus el contenido de КІ, que llega a la ALU por зи 
acceso derecho (véase la secuencia en las figuras 2.9 y 2.10). En 


* Se ha traducido la expresión buffer por memoria o registro auxiliar, 
pudiéndose encontrar en otros textos como registro tampón o registro 
intermedio. 
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Figura 2.8 REGISTROS 
Arquitectura de bus único. BUS (DE DESTINO) DEL RESULTADO 
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Figura 2.9 
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Figura 2.10 В 
Suma: el segundo registro R1 
pasa a ALU. REGISTROS 
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este momento, la entrada derecha de la ALU viene determinada 
por КІ, y la izquierda, por el registro auxiliar, que contiene el 
valor anterior de R0; acto seguido se realiza la suma en la 
UAL, y el resultado pasa a la salida de la misma (véase la 
figura 2.11, parte inferior derecha). Allí lo recoge el bus ünico y 
lo transporta hasta RO; en la práctica, esto significa que la 
entrada de RO está abierta y que el dato puede escribirse en su 
interior. El resultado de la suma ya está en RO, y la ejecución 
de la instrucción ha terminado. Obsérvese que el contenido de 
В! no se ha modificado; esto ilustra un principio general: la 
operación de lectura no modifica los contenidos de los registros 
ni de la memoria de lectura/escritura. 


BUS INTERNO DE DATOS 


BUS 
EXTERNO 


Figura 2.11 
El resultado que acaba de ge- 
nerarse va a RO. АС + R1 —» RO 


El registro auxiliar de la entrada izquierda de la ALU sirve 
para memorizar el contenido de RO, mientras el bus, que es 
ünico, recuérdese, se emplea para transferir otros contenidos; 
pero sigue habiendo un problema. 


EL PROBLEMA DE LA VELOCIDAD CRITICA 


La sencilla organización ilustrada en la figura 2.8 no funcio- 
naría correctamente. 


Pregunta: ¿En qué consiste el problema de la sincronización? 


Respuesta: El problema está en que el resultado producido por 
la ALU pasa al bus único y se propaga a todo lo largo del 
mismo, no únicamente en dirección de RO; en concreto, volverá 
a determinar la entrada derecha de la ALU y modificará el 


resultado, que irá a la salida unos pocos nanosegundos más 
tarde. En esto consiste el problema de la velocidad crítica. Es 
imprescindible aislar la salida de la ALU de su entrada 
(véase la figura 2.12). 
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Figura 2.12 
El problema de la velocidad crí- 
tica. 
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Para ello hay varias soluciones, siempre utilizando un regis- 
tro auxiliar. Este puede instalarse tanto a la salida como a la 
entrada de la ALU; por lo general, se coloca a la entrada, en 
este caso en su lado derecho. Ahora el sistema funcionará ya 
correctamente. Más adelante, en este mismo capitulo, veremos 
que si el registro ilustrado en el lado izquierdo se usa como 
acumulador (para instrucciones de un byte de longitud), necesi- 
tará, a su vez, uno auxiliar, como el representado en la figura 
2.13. 
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Figura 2.13 

Los dos registros auxiliares ne- 
cesarios (o registros tempora- 
les). REGISTROS 
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Organización interna del 280 


Los términos necesarios para entender los elementos inter- 
nos del microprocesador están ya definidos, de manera que 
podemos pasar a examinar con más detalle el Z80 propiamente 
dicho y a describir sus posibilidades. La organización interna 
del Z80 aparece en la figura 2.14, que constituye una descrip- 
ción lógica del dispositivo. Puede haber algunas otras intercone- 
xiones adicionales, pero no se han mostrado. Empezaremos a 
seguir el esquema a partir de la derecha. 

Lo primero que encontramos es la unidad aritmética y lógica 
(la ALU), fácilmente reconocible por su característica forma de 
V. El acumulador, del que ya hablamos en la sección anterior, 
situado en la entrada derecha de la ALU, se identifica como А. 
También vimos en esa sección que el acumulador debe contar 
con un registro auxiliar, identificado en la figura como ACT 
(acumulador temporal). También la entrada izquierda de la 
ALU dispone de un registro temporal, denominado TMP. El 
funcionamiento de la ALU se aclarará en la próxima sección, en 
la que describiremos la ejecución de instrucciones reales. 

En el 780, el registro de estado se llama Е, y aparece a la 
derecha del acumulador. Su contenido está básicamente deter- 
minado por la ALU, pero veremos que algunos de sus bits 
pueden también depender de otros módulos o de otros sucesos. 

Los registros acumulador y de estado aparecen duplicados e 
identificados respectivamente, À, A' y F, Е’; ello se debe a que 
el Z80 está quipado internamente de dos juegos de registros: 
A +F y A' + Е, aunque sólo puede usarse un juego a la vez; 
hay una instrucción especial para intercambiar los contenidos 
de A y F con los de A' y F'. Para simplificar las explicaciones, 
en la mayor parte de los esquemas ulteriores representaremos 
únicamente A y F; el lector deberá recordar que dispone, si lo 
desea, de la opción de pasar a los registros A’ y Е. 

La función de cada una de las banderas del registro F se 
describirá en el capítulo 3 (Técnicas básicas de programación). 

En el centro de la figura aparece un amplio bloque de regis- 
tros, en cuya parte superior se observan dos grupos idénticos. 
Cada uno de ellos dispone de seis registros, identificados por las 
letras В, С, D E, Н y І; se trata de los registros de tipo general 
de ocho bits del 780. Este tiene dos peculiaridades que lo 
diferencian del microprocesador general descrito al comienzo de 
este capítulo. 

En primer lugar, el Z80 dispone de dos bancos de registros, 
es decir, de dos grupos idénticos de seis registros cada uno. En 
un momento determinado, sólo pueden usarse seis registros, 
pero hay instrucciones especiales para intercambiar sus conteni- 
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Organización interna del 780. 


Figura 2.14 
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dos ente los dos grupos. Рог tanto, uno de ellos se comporta 
como memoria interna, mientras el otro funciona como bloque 
de registros; las posibles aplicaciones de esta peculiaridad se 
descibirán en el próximo capitulo. 

Para evitar confusiones, supondremos, a partir de ahora, 
que sólo hay seis registros de trabajo —B, С, D, E, H y L— e 
ignoraremos los del segundo grupo. 

E] símbolo MUX, que aparece en la parte superior de la 
memoria, es abreviatura de multiplexor; los datos procedentes 
del bus interno de datos pasan al registro seleccionado a través 
de ese elemento. En un momento determinado, sólo puede 
haber un registro conectado al bus interno. 

La segunda peculiaridad de estos seis registros, además de Й 
ser registros generales de ocho bits, es que disponen de una 
conexión con el “bus” de direcciones, y por eso se agrupan por 
pares. En efecto, los contenidos de B y C, por ejemplo, pueden 
entregarse simultáneamente al bus de direcciones de 16 bits, 
ilustrado en la parte inferior de la figura. Gracias a ello, los seis 
registros de este grupo pueden usarse tanto para almacenar 
datos de ocho bits como para guardar apuntadores de dieciséis 
bits para el direccionamiento de la memoria. 

El tercer grupo de registros, que en la ilustración aparece en 
el centro, bajo los dos anteriores, contiene cuatro registros de 
direcciones “puros”. Como en cualquier microprocesador, hay 
un contador del programa (PC) y un puntero de la pila (SP). 
Recuérdese que en el contador del programa está la dirección 
de la instrucción que ha de ejecutarse acto seguido. 

El puntero de la pila señala la parte superior de la pila de la 
memoria. En el caso del Z80, señala la última entrada real en la 
pila (en otros microprocesadores señala justo por encima de la 
última entrada). La pila crece hacia abajo, es decir, hacia las 
direcciones más bajas. 

Esto significa que el puntero debe disminuir cada vez que se 
introduce una nueva palabra en la pila. Y viceversa, cuando se 
extrae una palabra de la pila, el valor del puntero aumenta en 
uno. En el Z80, las operaciones de introducir y extraer siempre 
implican dos palabras al mismo tiempo, por lo que el valor del 
puntero disminuye y aumenta en esa medida. 

Faltan por describir dos registros del grupo central de cua- 
tro: se trata de dos registros índices llamados IX (registro 
índice X) e IY (registro indice Y), equipados ambos con un 
sumador especial representado como una ALU pequeña en 
forma de V, a la derecha de los mismos (seguimos en la figura 
2.14). Cuando se usa una instrucción indexada, por el bus de 
datos interno se mueve un byte especial llamado desplazamiento, 
que se suma a los contenidos de IX o IY. Hay instrucciones 


especiales que automáticamente suman este desplazamiento a 
IX o IY y generan una dirección; es lo que se llama indexación, 
y permite acceder cómodamente a cualquier bloque de datos 
secuenciales. Esta valiosa opción se describirá en el capitulo 5 
(Técnicas de direccionamiento). 

En la parte inferior del bloque central de registros, y a su 
izquierda, hay una casilla señalada “+ 1”; se trata de un suma- 
dor/restador que incrementa o decrementa automáticamente el 
contenido de cualquiera de los pares de registros SP, PC, BC, 
DE, o HL (los registros de direcciones “puros”) cada vez que 
pasan una dirección al bus interno de direcciones. Es un disposi- 
tivo indispensable para realizar los bucdes de programa automati- 
zados que describiremos en la próxima sección. Gracias a él 
se puede acceder cómodamente a posiciones sucesivas de la me- 
moria. 

Pasemos ahora al extremo izquierdo de la figura. Vemos un 
par de registros aislados identificados como Гу R. El I se llama 
registro de interrupción-dirección de página, y su función se des- 
cribirá en la sección dedicada a las interrupciones del capitulo 6 
(Técnicas de entrada y salida); se utiliza solamente en un caso 
especial en el que, como respuesta a una interrupción, se genera 
una llamada indirecta a una posición de memoria. El registro I 
que nos ocupa almacena la parte superior de la dirección indi- 
recta; la parte inferior de ésta la proporciona el dispositivo que 
provoca la interrupción. 

El registro R es el de refresco de la memoria, y sirve para 
refrescar automáticamente las memorias dinámicas. Como está 
asociado a la memoria dinámica, este registro suele montarse 
fuera del microprocesador; es un elemento muy cómodo, que 
reduce el soporte físico necesario para algunos tipos de memo- 
rias dinámicas. No la utilizaremos aquí con fines de programa- 
ción, puesto que se trata fundamentalmente de una característi- 
ca del soporte físico (véase Microprocessor Interfacing Techni- 
ques, donde se da una descripción detallada de las técnicas de 
refresco de la memoria); no obstante, puede utilizarse también 
como reloj del soporte lógico, por ejemplo. 

Pasemos ahora al extremo izquierdo de la ilustración, donde 
aparece la sección de control del microprocesador. Empezando 
por arriba, primero encontramos el registro de instrucción, IR, 
que contiene la instrucción que ha de ejecutarse (este registro no 
tiene nada que ver con el par “I, R” descrito más arriba). La 
instrucción se recibe de la memoria por medio del bus de datos, 
se transmite a lo largo del bus de datos interno y, por fin, se 
deposita en el registro de instrucción. Bajo éste se encuentra el 
decodificador, que envía señales al controlador-secuenciador y 
determina la ejecución de la instrucción dentro y fuera del 
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microprocesador. La sección de control genera y gobierna el bus 
de control ilustrado en la parte inferior de la figura. 

Los tres buses gobernados por el sistema —el de datos, el de 
direcciones y el de control — se prolongan fuera del microproce- 
sador por medio de las patillas de conexión del mismo; estas 
conexiones aparecen en el extremo derecho de la figura, y, como 
se ve, están aisladas del exterior por separadores. 

Ya hemos descrito todos los elementos lógicos del 780. Para 
empezar a escribir programas no es necesario conocer al detalle 
el funcionamiento del microprocesador, pero quien esté intere- 
sado por escribir códigos eficaces deberá entender de qué forma 
se ejecutan las instrucciones en su interior, porque la velocidad 
y el tamaño del programa dependen de la elección acertada de 
registros y técnicas; por tanto, examinaremos a continuación la 
ejecución de algunas instrucciones típicas en el interior del Z80, 
y veremos así el funcionamiento y la utilización de los registros 
internos y los buses. 


Formatos de las instrucciones 
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Las instrucciones del 780 aparecen en el capitulo 4 y pueden 
tener uno, dos, tres o cuatro bytes. Una instrucción especifica la 
operación que debe llevar a cabo el microprocesador. Desde un 
punto de vista simple, cualquier instrucción puede representarse 
mediante un código de operación seguido por un campo opcio- 
nal literal o de dirección, que consta de una o dos palabras. El 
código del campo de operación especifica la que ha de empren- 
derse. En términos informáticos estrictos, el código de opera- 
ción consta únicamente de los bits que especifican la operación 
que debe realizarse, con exclusión de los punteros que pudieran 
ser necesarios. Pero en el ámbito de los microprocesadores es 
útil llamar código de operación al conjunto de éste más los 
punteros. Este “código de operación generalizado” debe residir 
en una palabra de ocho bits para alcanzar la máxima eficacia 
(ya que éste es el factor limitante del número de instrucciones 
de que dispone el microprocesador). 

El 8080 utiliza instrucciones de uno, dos o tres bytes de 
longitud (véase la figura 2.15). Pero el Z80, que dispone de 
instrucciones adicionales indexadas, necesita un byte más. Los 
códigos de operación del 780 son, por lo general, de un byte de 
longitud, salvo en el caso de instrucciones especiales, de dos 
bytes. 

En algunas instrucciones, al código de operación debe seguir 
un byte de datos, de manera que la instrucción pasará a ser de 


| 
| 


dos bytes (salvo que haya indexación, lo que supone otro byte 
más). | . 

Hay ocasiones en que la instrucción exige la especificación 
de una dirección. Como éstas ocupan 16 bits —dos bytes—, la 
instrucción tendrá tres o cuatro bytes. 

Para cada byte de instrucción, la unidad de control tendrá 
que ejecutar una operación de tomar de la memoria, que tarda 
en cumplirse cuatro ciclos de reloj. Como es obvio, cuanto más 
corta sea la instrucción, tanto más rápida será la ejecución. 
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Formatos típicos de instruccio- 
nes. 


INSTRUCCIONES DE UNA PALABRA 


Estas instrucciones son, en principio, las más rápidas y las 
preferidas por los programadores. Una instrucción monopala- 
bra típica del Z80 sería: 


LD г, г 


que significa: “transferir el contenido del registro г а г”. Se 
trata de una operación registro a registro. Todos los micropro- 
cesadores necesitan esta clase de instrucciones, que permiten al 
programador transferir información de cualquiera a cualquiera 
de los registros de la máquina. Las que se refieren a registros 
especiales, como el acumulador u otros, pueden necesitar de un 
código de operación peculiar. 

Una vez ejecutada la instrucción anterior, el contenido de r 
será igual que el de г que, por su parte, по se habrá visto 
modificado durante la operación de lectura. 

Todas las instrucciones se representan internamente en for- 
mato binario, y la notación “LD r, r'” es simbólica o mnemó- 
nica y se llama representación de una instrucción en lenguaje 
ensamblador. Se trata, simplemente, de una codificación simbóli- 
ca más cómoda de utilizar que el verdadero código binario, que 
еп el caso que nos ocupa sería: OIDDDFFF (bits 0 a 7). 
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Esta representación sigue siendo parcialmente simbólica. 
Cada una de las letras D y F representa un bit binario; las tres D, 
“DDD”, corresponden a los tres bits que señalan el registro de 
destino. Bastan tres bits para seleccionar uno de los ocho regis- 
tros posibles; los códigos de dichos registros aparecen en la 
figura 2.16; el correspondiente al registro B, por ejemplo, es 
“000”, “001” el de C, etc. 

De la misma manera “FFF” son los tres bits que señalan el 
registro fuente. En este caso, la convención es que el registro r' 
sea la fuente y el r, el destino. Los bits de la representación 
binaria de una instrucción no se organizan a gusto del progra- 
mador, sino en función de las exigencias de la sección de con- 
trol del microprocesador, que debe decodificar la instrucción y 
ejecutarla. Por el contrario, la representación en lenguaje ensam- 
blador sí está pensada para la mayor comodidad del programa- 
dor. Podría argúirse que LD r, r' debería, en realidad, interpre- 
tarse como “transferir el contenido de r a г”. No obstante, se 
ha decidido convencionalmente, es decir, de forma arbitraria, 
que signifique lo contrario, para conservar la compatibilidad 
con la representación binaria. 
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Ejercicio 2.1: Escribir el código binario encargado de transferir el 
contenido del registro C al registro B. Consültense los códigos 
correspondientes a C y B en la figura 2.16. 


CODIGO] REGISTRO 


Figura 2.16 
Códigos de los registros. 


Otro ejemplo sencillo de instrucción de una palabra sería: 
ADD А, г 


que da lugar а la suma del contenido de un registro especifica- 
do (г) al acumulador A. Simbólicamente, la operación podría 
representarse mediante la expresión А = А +r. Como se verá 
en el capitulo 4, la representación binaria de esta instrucción es: 


10000 FFF 


siendo FFF el registro que debe añadirse al acumulador. Los 
códigos de los registros aparecen en la figura 2.16. 


Ejercicio 2.2: ¿Cuál sería el código binario de la instrucción encar- 
gada de sumar el contenido del registro D al acumulador? 


INSTRUCCIONES DE DOS PALABRAS 
ADD A,n 


Esta sencilla instrucción de dos palabras hace que se sume el 
contenido del segundo byte de la instrucción al acumulador. El 
contenido de la segunda palabra de la instrucción es lo que se 
llama un “literal”, es decir, un dato que se trata como grupo de 
ocho bits sin ningún significado particular. Podría ser un carác- 
ter o un dato numérico, pero lo importante es que su naturale- 
za es irrelevante a efectos de la operación. El código de la 
instrucción es: 


11000110 seguido por el byte de 8 bits “n” 


Se trata de una operación inmediata, cosa que, en la mayor 
parte de los lenguajes de programación, significa que la palabra 
o palabras siguientes contienen un fragmento de dato que no 
debe ser interpretado (en el sentido en que sí debe serlo un 
código de operación). De ello se deduce que la palabra o las 
dos palabras siguientes se tratan como literales. 

La unidad de control está programada para que “sepa” 
cuántas palabras tiene cada instrucción, de manera que siempre 
toma y ejecuta el número correcto en cada caso. Sin embargo, 
cuanto mayor sea el número de palabras de la instrucción, 
tanto más complicado le resultará decodificarlas a la unidad de 
control. 


INSTRUCCIONES DE TRES PALABRAS 
LD A, (nn) 


Esta instrucción necesita tres palabras. Significa: "cargar el 
acumulador con la dirección de la memoria especificada en los 
dos bytes siguientes de la instrucción". Como las direcciones 
tienen una longitud de 16 bits, ocupan dos palabras. La repre- 
sentación binaria sería: 
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00111010: 


Dirección inferior: 


8 bits para el código de operación. 
8 bits para la parte inferior de la 


dirección. 
Dirección superior: 8 bits para la parte superior de la 
dirección. 


Ejecución de instrucciones dentro del Z80 


Como ya hemos visto, todas las instrucciones se llevan a 
cabo en tres fases: TOMAR, DECODIFICAR y EJECUTAR. 
Daremos ahora algunas definiciones de interés. Cada una de las 
fases consume varios ciclos de reloj. El Z80 ejecuta cada una de 
ellas en uno o más ciclos lógicos que se llaman "ciclos de 
máquina"; el ciclo de máquina más breve dura tres ciclos de 
reloj. 

El acceso a la memoria lleva tres ciclos para cualquier 
operando y cuatro de reloj para la fase tomar. Dado que lo 
primero que hay que hacer con cualquier instrucción es tomar- 
la, las más rápidas necesitarán al menos cuatro ciclos de reloj, 
aunque la mayoría consumen más. 

Los ciclos de máquina se designan M1, M2, etc., y cada uno 
de ellos consume tres o más ciclos de reloj o “estados”, que se 
designan Ti, T2, etc. 


FASE TOMAR (FETCH) 


La fase de TOMAR la instrucción se lleva a cabo durante 
los primeros tres estados del ciclo de máquina МІ, denomina- 
dos T1, T2 y T3. Estos tres estados son comunes a todas las 
instrucciones del microprocesador, porque todas deben tomarse 
antes de pasar a su ejecución. El mecanismo de TOMAR es el 
siguiente: 


T1: SALIDA del PC 


El primer paso es la presentación a la memoria de la direc- 
ción de la instrucción siguiente, dirección que figura en el conta- 
dor del programa (PC). Como ocurre en la fase tomar de todas 
las instrucciones, se empieza por colocar el contenido del PC en 
el bus de direcciones (véase la figura 2.17). En este momento se 
presenta una dirección a la memoria, donde es interpretada por 
el correspondiente decodificador de direcciones para seleccionar 
la posición de memoria correcta. Antes de que el contenido de 


BUS DE DATOS 


пъ-г-жс» 


REG. INST. 
DECODIFI- 
CADOR 


CONTRO- 
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CIADOR 
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Figura 2.17 
Tomar la instrucción: el PC se 
dirige a la memoria. 


SEÑALES 
DE CONTROL 


esa posición llegue a las patillas de salida de la memoria, 
conectadas al bus de datos, deben transcurrir varios cientos de 
nanosegundos (un nanosegundo, ns, es igual а 107° segundos). 
En diseño informático, es norma universal utilizar el tiempo de 
lectura en la memoria para realizar una operación dentro del 
microprocesador. Esta operación es el incremento del contador 
del programa: 


T2: РС= РС +1 


Mientras se lee la memoria, el contenido del PC se incre- 
menta en 1 (véase la figura 2.18). Al final del estado T2 el 
contenido de la memoria ya está disponible y puede transferirse 
dentro del microprocesador: 


T3: INST en IR 


А 
[9] 
A (CA À ХҚ > sus ve DATOS 
A, $ n 
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EREE | 
FICADOR 
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е. [е] Е 
CONTRO- 
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= | 
А р 
ZZ, 2 X 
16] L 
A À L^ BUS DE DIRECCIONES 


SENALES 
DE CONTROL 


Figura 2.18 
Incremento del РС. 


67 


3 


a 


FASES DECODIFICAR Y EJECUTAR 


Durante el estado T3, la instrucción leida en la memoria se 
deposita en el bus de datos y se transfiere al registro de instruc- 
ción del 780; la decodificación tiene lugar а partir de este 
punto. 


DECODI- 
FICADOR 


CONTRO- 
LADOR 
SECUEN- 
CIADOR 


Г. BUS DE DIRECCIONES 


Figura 2.19 
La instrucción llega a IR proce- 
dente de la memoria. 


SENALES 
DE CONTROL 


Hay que observar que М1 siempre debe alcanzar un estado 
T4; en efecto, una vez depositada la instrucción en IR durante 
T3, hay que decodificarla y ejecutarla, lo que exige, al menos, un 
nuevo estado de máquina T4. 

Algunas instrucciones exigen otro estado adicional T5 de 
МІ, que el procesador salta en la mayor parte de los casos. Si 
la instrucción exige dos o más ciclos M1, M2, etc., se produce 
directamente la transición desde el estado T4 de M1 al Т1 de 
M2. Veremos a continuación un ejemplo, que estudiaremos con 
ayuda de las secuencias internas que refleja la tabla de la figura 
2.27 (estas tablas no se han publicado todavía para el Z80, por 
lo que usamos en su lugar las correspondientes al 8080; dan 
una visión en profundidad de las fases que sigue la ejecución de 
una instrucción). 
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LD D, C 


Esta instrucción del Z80 corresponde a la MOV rl, r2 del 
8080, que recoge en su primera línea la tabla de la figura 2.27. 
Da la casualidad que el registro de destino de este ejemplo 
se llama también D. La transferencia se ilustra en la figura 2.20. 


Figura 2.20 
Transferencia de C a D. 


Figura 2.21 
El contenido de C se deposita 
en TMP. 


D с 
00610001 10001000 


ANTES 


Ганна] 


10001000 10001000 


DESPUES 


La instrucción, que ya hemos descrito en la sección anterior, 
lleva el contenido del registro C, llamado “c”, al registro D. 

Los tres primeros estados del ciclo МТ se emplean en traer 
la instrucción de la memoria. Al término de T3 ya está en el 
registro de instrucción, IR, desde el que será decodificada (véase 
la figura 2.19). 

Durante T4: (FFF) » TMP. 

El contenido de C se deposita en TMP (véase la figura 2.21). 

Durante 15: (TMP) > DDD. 

El contenido de de TMP se deposita en D (véase la figura 
2.22). 
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DE CONTROL 


La fase de ejecución ya ha terminado. El contenido del 
registro C ha pasado al registro D, que constituia su destino de 
acuerdo con lo especificado, y de esta forma acaba la ejecución 
de la instrucción. Los ciclos de máquina M2, M3, M4 y M5 no 
son necesarios, y la operación completa se ha realizado en M1. 

Es fácil calcular el tiempo real que tarda en ejecutarse la 
instrucción. En el Z80 normal, la duración de un estado es la 
del reloj: 500 ns. Esta instrucción requiere el transcurso de 
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Figura 2.22 
El contenido de TMP se depo- 
sita en D. 


SEÑALES 
DE CONTROL 


cinco estados, es decir, 5 x 500 = 2500 ns = 2.5 us. Con un 
reloj de 400 ns, sería 5 x 400 = 2000 ns = 2.0 ps. 


Pregunta: ¿Por qué esta instrucción necesita dos estados —T4 y 
T5— para transferir el contenido de C a D en lugar de sólo 
uno? Primeramente lleva el contenido de C a TMP, y a 
continuación el de éste a D. ¿No sería más fácil pasar el 
contenido de C a D directamente y en un solo estado? 


Respuesta: Eso sería imposible, debido a la disposición de los 
registros internos del Z80. Todos los registros forman parte 
de una única memoria RAM de lectura/escritura construida 
dentro del microcircuito integrado que constituye el micro- 
procesador. Como la RAM tiene una sola puerta, ünicamen- 
te puede direccionarse o escogerse una palabra en un mo- 
mento determinado, de tal manera que es imposible leer y 
escribir simultáneamente en dos posiciones diferentes de la 
memoria. Por eso hay que consumir dos ciclos de RAM, 
uno para leer los datos y almacenarlos en el registro tempo- 
ral TMP y otro para escribirlos en el registro final de 
destino (D, en este caso). Es una insuficiencia de diseño 
comün prácticamente a todos los microprocesadores mono- 
líticos. Para resolver el problema sería necesario una RAM 
de doble puerta. De todas formas, la limitación no es intrín- 
seca a los microprocesadores, y no se encuentra en los 
dispositivos “a rebanadas"*; es consecuencia de la tendencia 
dominante hacia el aumento de la densidad lógica de los 
circuitos integrados, y podría solucionarse en el futuro. 
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* Bit slice: “a rebanadas”: diversos chips que forman una CPU (suele estar 
basada en chips de 4 bits; por ejemplo, la familia 2900, formando UAL, de 
n x 4 bits). 
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EJERCICIO IMPORTANTE 


Llegados a este punto, es muy recomendable que el lector 
repase la secuencia de esta sencilla instrucción antes de pasar a 
otras más complicadas. Vuelva para ello a la figura 2.14, reúna 
unos pocos “simbolos” pequeños —cerillas, clips, etc.— y des- 
plácelos sobre la ilustración para simular el flujo de datos de 
los registros a los buses. Por ejemplo: deposite un símbolo en el 
PC; en Tl, dicho símbolo recorrerá el bus direcciones hacia la 
memoria; continúe la simulación por el mismo procedimiento 
hasta que considere que domina los movimientos que tienen 
lugar entre buses y registros; en ese momento estará en condi- 
ciones de pasar a las siguientes instrucciones, que serán сада 
vez más complejas. 


ADD A, r 


La instrucción significa: «sumar el contenido del registro r 
(especificado mediante un código binario FFF) al acumulador 
(A) y depositar el resultado en dicho acumulador”. Se trata de 
una instrucción implícita, es decir, que no hace referencia explí- 
cita a un segundo registro, sino sólo a uno llamado r, lo que 
implica que el otro afectado por la operación es el acumulador. 
Este, cuando se usa en una instrucción implícita, se toma como 
origen y como destino. Como resultado de la operación, los 
datos se almacenan en el acumulador. La ventaja de esta ins- 
trucción implícita es que el código de operación completo sólo 
tiene ocho bits de longitud y no necesita más que un campo de 
registro de tres bits para la especificación de r. Se trata, pues, de 
una forma rápida de sumar. 

El sistema dispone de otras instrucciones implícitas que se 
refieren a registros especializados. Son ejemplos de instrucciones 
de esta clase las PUSH (empujar) y POP (extraer), que movili- 
zan información entre el extremo superior de la pila y el acu- 
mulador al tiempo que actualizan el puntero de la pila (SP). 
Implicitamente manipulan el registro SP. 

A continuación analizaremos en detalle la ejecución de 
ADD A, r. La instrucción precisa dos ciclos de máquina, Ml y 
M2. Como es usual, durante los tres primeros estados de М1 se 
trae la instrucción de la memoria y se deposita en el registro IR. 
Al principio de T4 se decodifica y puede ya ejecutarse. Supon- 
dremos aquí que el registro В se suma al acumulador. El código 
de la operación es 10000000 (el código del registro B es 000). La 
instrucción 8080 equivalente a ésta es ADD r. 


Т4: (FFF) > TMP; (А) > АСТ 


Figura 2.23 
Dos transferencias que ocurren 
simultáneamente, 
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Como se ve, tienen lugar simultáneamente dos transferen- 
cias. Primero se transfiere a TMP el registro fuente especificado 
(B en este caso), lo que significa que se coloca a la entrada de la 
ALU (véase la figura 2.23). Al mismo tiempo se transfiere el 
contenido del acumulador al acumulador temporal ACT. Si 
observa la figura 2.23, comprenderá que los dos movimientos 
pueden realizarse en paralelo, porque siguen caminos diferentes 
dentro del sistema. El paso de B a TMP sigue el bus interno de 
datos, mientras que el de A a ACT se efectúa a través de un 
breve camino interior independiente del mencionado bus de 
datos. La realización simultánea de los dos movimientos supone 
ahorro de tiempo. En este momento las entradas derecha e 
izquierda de la ALU se encuentran correctamente determina- 
das; la primera, por el contenido del registro B, y la segunda, 
por el del acumulador. El siguiente paso es, pues, la adición. 
Cabría esperar que se llevase a cabo durante el estado T5 de 
Mi, pero dicho estado no se utiliza, y la suma queda sin 
realizar. Se pasa al ciclo M2, pero durante el estado T! tampo- 
co ocurre nada. La adición no tiene lugar hasta el estado T2 de 
M2 (véase ADDren la figura 2.27): 


T2 de M2: (ACT) + (TMP) » А 
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El contenido de ACT se suma al de TMP, y el resultado se 
deposita en el acumulador (figura 2.24). La operación ya está 
completa. 


Pregunta: ¿Por qué se difiere el término de la adición hasta el 
estado T2 del ciclo de máquina M2 en lugar de ejecutarla en 
el estado T5 de M1? (Es difícil responder a esta pregunta sin 


атолу 


tener ciertos conocimientos sobre diseño de CPU; básica- 
mente, se trata de una técnica de diseño de CPU sincroniza- 
da con el reloj. Trataremos a continuación de explicar lo 
que ocurre.) 


Respuesta: Es un “truco” de diseño, común a casi todas las 
CPU, que se llama “solapamiento toma/ejecución". En esen- 
cia, se trata de lo siguiente: como se observa en la figura 
2.23, la ejecución de la adición propiamente dicha sólo pre- 
cisa de la ALU y del bus de datos y, en particular, no 
necesita acceder a la RAM (bloque de registros). La unidad 
de control “sabe” que los tres estados ejecutados a continua- 
ción de cualquier instrucción son siempre T1, T2 y T3 del 
ciclo de máquina M1 de la instrucción siguiente. Al repasar 
la ejecución de estos tres estados se aprecia que, para ello, 
basta acceder al contador del programa PC y utilizar el bus 
de direcciones. Acceder al contador del programa supone 
acceder a los registros de RAM (esto explica por qué no 
puede recurrirse a la misma idea en la instrucción LD, г, г). 
Por tanto, es posible trabajar simultáneamente en las zonas 
sombreadas de las figuras 2.17 y 2.24. 
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Figura 2.24 
Final de ADD r. 


El bus de datos se utiliza en ТІ de МІ para transportar 
información sobre el estado, pero no puede utilizarse para la 
adición que deseamos ejecutar. Por ello es preciso esperar hasta 
el T2 antes de sumar, y eso es lo que ocurre en la tabla. La 
ventaja del mecanismo que acabamos de explicar la veremos a 
continuación: supongamos que procedemos de lo que sería la 
forma normal y ejecutamos la suma durante el estado T5 del 
ciclo de máquina МТ; la duración de la instrucción ADD sería 
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Figura 2.25 
Solapamiento TOMAR-EJE- 
CUTAR durante T1-T2. 
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de esta forma: 5 x 500 = 2 500 ns. Con el solapamiento, la 
instrucción siguiente comienza al término del estado T4, y la 
“astuta” unidad de control aprovechará el estado T2 de esta 
segunda instrucción para terminar la suma, pero sin que ello 
afecte a dicha siguiente instrucción. En la tabla, T2 se presenta 
como parte de M2, que, conceptualmente, es el segundo ciclo de 
máquina de la adición, aunque en realidad T2 pertenece al ciclo 
M1 de la instrucción siguiente. Para el programador, la instruc- 
ción ADD consume únicamente cuatro estados, es decir, 
4 x 500 = 2000 ns, en lugar de los 2500 que emplearía el 
tratamiento convencional. La velocidad ha mejorado en 500 ns, 
equivalentes a un 20 por 100. 

La técnica del solapamiento se describe en la figura 2.25, y 
se utiliza siempre que mediante ella sea posible incrementar la 
velocidad de trabajo aparente del microprocesador. Natural- 
mente, no en todos los casos se puede solapar, porque los buses 
y dispositivos necesarios han de estar disponibles, sin que ello 
suponga conflicto. La unidad de control “sabe” en todo mo- 
mento si el solapamiento es o no posible. 
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Pregunta: ¿Sería posible llevar más lejos este recurso y utilizar 
también el estado T3 de M2 para ejecutar una instrucción más 
larga? 


Para clarificar el mecanismo interno de secuenciación es 
aconsejable examinar la figura 2.27, que recoge en detalle la 
ejecución de las instrucciones del 8080. El Z80 tiene todas las 
instrucciones del 8080 y algunas más. Si se ha reproducido aqui 
es porque contribuye a esclarecer el funcionamiento interno de 
este microprocesador. Los apéndices F y G recogen las equiva- 
lencias entre el Z80 y el 8080. Pasemos ahora a estudiar una 
instrucción más compleja: 


ADD A, (HL) 


El código de operación es 10000110. La instrucción significa 
“Sumar al acumulador el contenido de la posición de memoria 
(HL)”. Esta posición se especifica de una forma un tanto extra- 
ña, porque es aquella cuya dirección está contenida en los 
registros H y L. La instrucción supone que estos dos registros 


Figura 2.26 


Abreviaturas Intel. 


NOTAS: 


1. El primer ciclo de memoria (M1) siem- 
pre es tomar una instrucción; durante el 
mismo se toma el primer (a veces, el único) 
byte del código de operación. 

2. Si la entrada READY de la memoria no 
está en alto durante T2 de cada ciclo de 
memoria, el procesador entrará en estado de 
espera (TW) hasta que se detecte en alto. 


3. Los estados T4 y T5 están disponibles, 
si son necesarios, para operaciones comple- 
tamente internas a la CPU. El contenido del 
bus interno durante T4 y T5 está a disposi- 
ción del bus de datos, aunque esto sólo sir- 


gistros pr = B (registros B y C) o pr=D 
(registros D y E). 

5. Estos estados se saltan. 

6. Subciclos de lectura en memoria; pue- 
den leerse palabras de instrucción o de da- 
to. 


7. Subciclo de escritura en memoria. 


8. La señal READY no es necesaria duran- 
te los subciclos segundo y tercero (M2 y 
M3). La señal HOLD se acepta durante M2 
y M3. La señal SYNC no se genera durante 
M2 y M3. Durante la ejecución de DAD ha- 
cen falta M2 y M3 para la suma de un par 
de registros internos; no se referencia la me- 
moria. 

9. Los resultados de estas instrucciones 
aritméticas, lógicas o de rotación no se 
mueven al acumulador (A) hasta el estado 
T2 del siguiente ciclo de instrucción, de 
manera que А se carga mientras se toma la 
instrucción siguiente, Este solapamiento de 
operaciones acelera la velocidad de proceso. 


10. Si el valor de los 4 bits menos signifi- 
cativos del acumulador es superior a 9 o si 
el bit auxiliar de arrastre se ha fijado, se su- 
ma 6 al acumulador. Si el valor de los 4 bits 
más significativos del acumulador es ahora 
superior a 9 o si el bit de arrastre se ha fija- 
do, se suma 6 a los 4 bits más significativos 
del acumulador. 


11. Esto representa el primer subciclo 
(traer la instrucción) del siguiente ciclo de 
instrucción. 


Por cortesía de Intel Corporation 


12. Si se satisface la condición, el conte- 
nido del par de registros WZ se lleva a las 
líneas de salida (Ao.15) en lugar del conte- 
nido del contador del programa PC. 

13. Si la condición no se satisface, se sal- 
tan los subciclos M4 y Mb; el procesador 
pasa inmediatamente a la fase traer (M1) 
del siguiente ciclo de instrucción. 


14. Si la condición no se safisface, se sal- 
tan los subciclos M2 y M3; el procesador 
pasa inmediatamente a la fase traer (M1) 
del siguiente ciclo de instrucción. 

15. Subciclo de lectura de la pila. 


16. Subciclo de escritura de la pila. 


ve con fines de verificación. "X" significa 17. CONDICION CCC 
que el estado está presente, pero sólo se NZ — no cero (Z = 0) 000 
utiliza en operaciones internas, como la de- 2 cero (2 = 1) 001 
codificación de instrucciones. NC — sin acarreo (СУ = 0) 010 
4. Sólo pueden especificarse pares de re- С - acarreo (CY = 1) 011 


PO — paridad impar (P =0) 100 

PE — paridad par (P = 1) 101 

P — más ($ = 0) 110 

M — menos (5 = 1) 111 
18. Subciclo E/S: el código de selección 
de la puerta de E/S de 8 bits se duplica en 
las líneas de dirección 0-7 (Ao.;) y 8-15 
(Ав.т5). , 
19. Subciclo de salida. 


20. El procesador permanecerá inactivo en 
estado de detención hasta que acepte una 
indicación de interrupción, reinicio (RESET) 
o mantenimiento (HOLD). Si se acepta esta 
última, la CPU pasa a función de mantener 
у, al final de la misma, de nuevo a estado 
de detención. Si se acepta el reinicio, el 
procesador empieza la ejecución por la po- 
sición de memoria cero. Tras aceptar una 
interrupción, el procesador ejecuta la ins- 
trucción introducida en el bus de datos (por 
lo general, una instrucción de vuelta a ce- 4. 
ro). 


FFF o DDD 
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Figura 2.27 | 
Formatos de las instrucciones 
Intel. 
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especiales (HL) se han cargado antes de la ejecución de la 
misma, de manera que sus contenidos de 16 bits especificarán la 
dirección de la memoria en que residen los datos. Estos datos 
se sumarán acto seguido al acumulador, que será el lugar en 
que se deposite también el resultado. 

Esta instrucción se ha creado para garantizar la compatibili- 
dad entre el antiguo 8008 y su sucesor, el 8080. El 8008 no 
disponía de direccionamiento directo de la memoria, por lo que 
para acceder a ésta se cargaban dos registros, H y L, y se 
ejecutaba una instrucción que se refiriese a ellos, como ADD A, 
(НІ). Hay que insistir en que el 8080 y el 780 no tienen la 
limitación de direccionamiento de la memoria del 8008, sino 
que cuentan con acceso directo a la misma; por tanto, la 
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posibilidad de trabajar con los registros H y L se convierte en 
una ventaja, y no en un inconveniente, como era en el 8008. 

Sigamos ahora su ejecución (la instrucción se llama ADD M 
en el 8080, y ocupa el número 16 en la figura 2.27). Como es 
habitual, se utilizan los estados T1, T2 y T3 de M1 para tomar 
la instrucción. Durante el estado T4 se transfiere el contenido 
del acumulador a un registro auxiliar ACT, y se determina la 
entrada izquierda de la ALU. 

Ahora es preciso acceder a la memoria para obtener el 
segundo byte de datos que debe sumarse al acumulador. La 
dirección de este byte está contenida en H y L, cuyo contenido, 
deberá, por tanto, transferirse al bus de direcciones y, mediante 
éste, a la memoria. 


(ACT | t TMe ALU-H, CY 
AACTISTMPISCY-ALU 
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Durante el ciclo de máquina M2 se lee HL y su contenido 
se deposita en el bus de direcciones exactamente igual que en 
otras instrucciones se depositaba el contenido del PC. Ya se ha 
indicado que durante Т1 se lleva el estado al bus de datos, 
aunque aquí no se hará uso del mismo. Simplificando las cosas, 
hacen falta dos estados: uno, para que la memoria lea los datos, 
y otro, para que los datos pasen al TMP de la entrada derecha 
de la ALU. 

Ya están, pues, determinadas las dos entradas de la ALU. 
La situación es idéntica a la que ya hemos visto en la anterior 
instrucción, ADD A, r; por tanto, no hay más que efectuar la 
suma. Se recurre también a la técnica de solapamiento traer/eje- 
cutar y, en lugar de hacer la suma en el estado T4 de M2, se 
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retrasa la ejecución hasta el T2 de M3. Como se ve en la figura 
2.27, durante T2: ACT + TMP - А. Por fin se realiza la suma 
de los contenidos de ACT y ТМР, y el resultado se deposita en 
el acumulador A. 


Pregunta: ¿Cuál es el tiempo aparente de ejecución de la instruc- 
ción para el programador? Si se utilizase un reloj de 2,5 Mhz, 
¿sería de 3,6 us o de 2,8 us? 


Examinaremos ahora una instrucción más complicada con 
direccionamiento directo de la memoria que utiliza dos registros 
invisibles, W y Z: 
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Figura 2.28 
Transferencia del contenido de 
HL al bus de direcciones. 


Figura 2.29 
LD A, (DIRECCION) es una 
instrucción de tres palabras. 
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CONTROL 
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LD A, (пп) 


El código de operación es 00111010. La instrucción equiva- 
lente del 8080 es ГОА addr. Como es habitual, se emplean los 
estados T1, T2 y ТЗ de М1 para traer la instrucción de la 
memoria. También se utiliza T4, pero no puede describirse 
ningún resultado visible, porque durante ese estado se decodifica 
la instrucción. En ese momento, la unidad de control averigua 
que debe traer los siguientes dos bytes de la instrucción para 
obtener la dirección a partir de la que debe cargarse el acumula- 
dor. El efecto de esta instrucción es cargar el acumulador con los 
contenidos de memoria cuyas direcciones especifican los bytes 2 
y 3 de la misma. Obsérvese que es preciso el estado T4 para 
decodificar la instrucción; esto podría considerarse una pérdida 
de tiempo, porque para esa labor basta una parte del estado, 
pero esa es la filosofía de la lógica sincronizada por reloj. Dado 
que internamente se utilizan microinstrucciones para llevar a 
cabo la decodificación y la ejecución, ése es el precio que hay que 
pagar a cambio de las ventajas de la microprogramación. La 
estructura de la instrucción aparece en la figura 2.29. 


CODIGO 

н: ША (ВІ) :DE OPERACION 
ntl: (B2) | DIRECCION 

| DIRECCION DETO ES 


(B3) 


Figura 2.30 
Situación previa a la ejecución 
de LD A. 


Figura 2.31 
Situación posterior a la ejecu- 
ción de LD A. 


A continuación se toman los dos bytes siguientes de la 
instrucción, que especificarán una dirección (véase la figura 
2.30). 

El resultado de la instrucción se ilustra en las figuras 2.30 
y 2.31 de esta misma página. 


(hex) 
А (НЕХ 


1111010] 1004 00111010 ША — (3A) 


0000000 №00000ба 101: 00000010 lc (02) 
PC 102 00010000 (НЕХ) (10) 


MEMORIA 


: 00111010 
: 00000010 
00010000 


REGISTROS 
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000000010 0000011 
РС 1 00001111 
REGISTROS MEMORIA 


La unidad de control (pero no el programador) tiene acceso 
a dos registros especiales internos del Z80: son los “W” y *Z”, 
ilustrados en la figura 2.28. 

Segundo ciclo de máquina M2: Como es habitual, los prime- 
ros dos estados, T1 y T2, se utilizan para tomar los contenidos 
de la posición de memoria PC. Durante T2 se incrementa el 
contador del programa. Hacia el final de este estado, los datos 
de la memoria quedan disponibles, y aparecen en el bus de 
datos. Al final de T3, la palabra que había sido tomada de la 
dirección de memoria PC (B2, segundo byte de la instrucción) 
está disponible en el bus de datos, que lo conduce a un registro 
temporal, Z: B2 в Z (véase la figura 2.32). 

Ciclo de máquina M3: Una vez más se deposita el PC en el 
bus de direcciones; se incrementa y, por último, se lee en la 
memoria, y se deposita en el registro W del microprocesador el 
tercer byte B3. En este punto, al final del estado T3 de M3, los 
registros internos W y Z contienen B2 y B3, es decir, la direc- 
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Figura 2.32 
El segundo byte de la instruc- 
ción llega a Z. 
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ción completa de 16 bits originalmente contenida en la memoria 
en forma de dos palabras tras la instrucción. La ejecución 
puede ya completarse: W y Z contienen una dirección que debe 
enviarse a la memoria para extraer el dato; estas operaciones se 
llevan a cabo en el siguiente ciclo de memoria. 

Ciclo de máquina M4: W y Z envían a la memoria la 
dirección de.16 bits a través del bus de direcciones; al término 
del estado T2, quedan disponibles los datos correspondientes al 
contenido de la posición de memoria especificada, que se depo- 
sita en А al final del estado T3, con lo que concluye la ejecución 
de la instrucción. 

Lo que acabamos de ver ilustra el uso de una instrucción 
inmediata, que precisa de tres bytes para almacenar una direc- 
ción explicita de dos bytes. Consume cuatro ciclos de memoria, 
porque necesita ir tres veces a ésta para tomar los tres bytes de 
las tres palabras y otra más para traer el dato especificado por 
la dirección. Se trata de una instrucción larga, pero, a la vez, es 
básica para cargar el acumulador con contenidos especificados 
que residen en una posición de memoria conocida. Obsérvese 
que la instrucción exige el empleo de los registros W y Z. 


Pregunta: ¿Podría esta instrucción haber utilizado otros registros 
internos al sistema diferentes de W y Z? 


Respuesta: No. Si esta instrucción utilizase otros registros, los 
H y L por ejemplo, modificaría sus contenidos, que habrian 
perdido al término de la ejecución. En un programa se 
supone que una instrucción no modificará más registros que 
los que emplea explicitamente. Si la instrucción carga el 
acumulador, no ha de destruir el contenido de ningún otro 
registro; por ello es imprescindible crear dos registros adi- 
cionales, W y Z, para uso interno de la unidad de control. 


Pregunta: ¿Podría haberse utilizado el PC en lugar de W y Z? 


Respuesta: De ninguna manera. Eso resultaría suicida (queda 
para el lector la demostración del porqué). 


Veremos ahora un nuevo tipo de instrucción, llamada de 
bifurcación o salto, que modifica la secuencia de ejecución de las 
instrucciones del programa. Hasta ahora hemos supuesto que 
las instrucciones se ejecutaban unas tras otra, pero veremos que 
hay algunas que permiten al programador saltar a un punto del 
programa diferente o, en términos prácticos, saltar a otra zona 
de la memoria que alberga el programa o a otra dirección. Una 
instrucción de esta naturaleza es: 


JP nn 


La instrucción aparece en la línea 18 de la figura 2.27 como 
“JPM addr”, y su ejecución puede seguirse en la misma tabla. 
Se trata también de una instrucción de tres palabras. La prime- 
ra es el código de operación 11000011. Las dos siguientes con- 
tienen la dirección de 16 bits a la que debe saltarse. Conceptual- 
mente, el efecto de la instrucción es sustituir el contenido del 
contador del programa por los 16 bits que siguen al código de 
operación “JUMP” (saltar). "Еп la práctica, y por razones de 
eficacia, la cuestión se aborda de forma algo diferente. 

Como antes, los tres primeros estados de M1 se emplean en 
tomar la instrucción. En T4 se decodifica ésta, y no se registra 
ningún otro suceso (X). Los dos siguientes ciclos de máquina se 
emplean en tomar los bytes B2 y B3 de la instrucción. En M2 
se toma B2 y se deposita en el registro interno Z. Como en la 
suma, el procesador dará los dos pasos siguientes mientras 
realiza la siguiente operación de tomar. Dichos dos pasos se 
ejecutan en lugar de los T1 y T2 habituales de la instrucción 
siguiente, como veremos a continuación. 

Esos dos pasos son: sacar WZ у (WZ) + | » PC. En otras 
palabras, durante la siguiente operación de tomar se utiliza el 
contenido de WZ en lugar del contenido del PC. La unidad de 
control tendrá en cuenta que se está dando un salto, y ejecutará 
el principio de la instrucción siguiente de forma distinta. 

El efecto de esos dos estados adicionales es el siguiente: la 
dirección depositada en el bus de direcciones del sistema será la 
contenida en W y Z, es decir, se tomará de la dirección conteni- 

da en W y Z, lo que, efectivamente, supone dar un salto. 


Además, el contenido de WZ se incrementa en 1 y se deposita 
en el contador del programa, para que la siguiente instrucción 
se tome correctamente utilizando el PC de la forma habitual. El 
efecto es, pues, el buscado. 


Pregunta: ¿Por qué se usan los registros intermedios W y Z en vez 
de cargar directamente el contenido del PC? 


Respuesta: El PC no puede usarse. Si cargásemos la parte. 
inferior del mismo (PCL) con B2 en lugar de usar Z, 
destruiríamos el PC, y sería imposible tomar B3. 


Pregunta: ¿Sería posible utilizar sólo Z en lugar de W y Z? 


Respuesta: Sí, pero el proceso sería más lento. Se podría cargar 
Z con B2, tomar a continuación B3 y depositarlo en la 
mitad superior del PC (PCH), pero ello obligaría a transfe- 
rir Z al PCL antes de utilizar el contenido del PC, y el 
avance sería así más lento. Por eso se trabaja con W y Z. 
Además, y para ganar tiempo, W y Z no se transfieren al 
PC, sino que se llevan directamente al bus de direcciones 
para tomar la instrucción siguiente. Comprender este punto 
es crucial para comprender la ejecución eficaz de instruccio- 
nes dentro del microprocesador. 


Pregunta (sólo para el lector con cierta experiencia): ¿Qué ocu- 
rriría si se produjese una interrupción al final de M3? (Si la 
ejecución de la instrucción se suspendiera en ese punto, el 
contador del programa señalaria la instrucción siguiente al 
salto, y la dirección de éste, contenida en W y Z, se perde- 
ría.) 


La averiguación de la respuesta constituirá un ejercicio pro- 
vechoso para el lector con cierto nivel de conocimientos. 

Las desripciones detalladas de instrucciones típicas que he- 
mos visto habrán clarificado la función de los registros y de los 
buses internos. Una segunda lectura de todo lo anterior contri- 
buirá a afianzar los conocimientos sobre las operaciones que 
tienen lugar en el interior del 780. 


EL MICROCIRCUITO INTEGRADO 780 


Para completar este capítulo estudiaremos las señales del 
circuito integrado Z80. Conocer estas señales no es necesario 
para programar, y el lector no interesado por los dealles del 


soporte físico puede saltarse esta parte con toda tranquilidad. 
La disposición de las patillas del Z80 se ilustra en la figura 2.33.. 
El bus de direcciones y el de datos, a la derecha del dibujo, 
cumplen su función habitual, ya descrita al principio del capítu- 
lo. Estudiaremos aquí las señales del bus de control, que apare- 
cen en el lado izquierdo de la misma figura. 
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Figura 2.33 
Patillas de salida del „P del MASA +5V 
280. ALIMENTACION 


Esas señales de control se dividen en cuatro grupos, y co- 
menzaremos su descripción por la parte superior del dibujo. 

La entrada del reloj es Ф. El 780 necesita una resistencia de 
330 ohmios, que se conecta a las entradas Фу a la toma de 5 
voltios; a 4 MHz hace falta un amplificador externo para el 
reloj. 

Las dos señales del bus de control, BUSRQ y BUSAK, 
sirven para desconectar el Z80 de sus buses, y son utilizadas, 
sobre todo, por el DMA (acceso directo a la memoria, direct 
memory access) aunque son igualmente accesibles por otros 
procesadores del sistema. BUSRQ es la señal de solicitud de 
bus; en respuesta a ella, el Z80 pone sus bus de direcciones y de 
datos y las sefiales de control de la salida de triple estado en 
situación de alta impedancia al término del ciclo de máquina en 
curso. BUSAK es la sefial de reconocimiento emitida por el Z80 
una vez que los buses han pasado a situación de alta impe- 
dancia. 


85 


Seis de las señales de control del Z80 están relacionadas con 
su estado interno o con el secuenciamiento: 

INT y NMI son dos señales de interrupción. INT es la 
solicitud de interrupción habitual (las interrupciones se tratarán 
en el-capítulo 6). A la línea INT pueden conectarse varios 
dispositivos de entrada y salida. Cuando se presenta en la 
misma una solicitud de interrupción —y si el biestable interno 
de permisión de interrupciones (IFF) se encuentra en el estado 
adecuado—, el Z80 la acepta (en el supuesto de que la entrada 
BUSRQ no esté activada); a continuación emitirá una señal de 
reconocimiento IORQ, que se produce durante el estado М1. El 
resto de la secuencia de acontecimientos se describe en el capí- 
tulo 6. 

NMI es la interrupción no enmascarable, aceptada siempre 
por el Z80, que salta a la posición hexadecimal 0066 (supo- 
niendo, igualmente, que BUSRQ no esté activada). Se describe 
en el capítulo 6. 

WAIT (esperar) es la sefíal que se utiliza para sincronizar el 
Z80 con los dispositivos de memoria o de entrada y salida de 
baja velocidad. Cuando se activa, la señal indica que la memo- 
ria o el dispositivo no están todavíà listos para la transferencia 
de datos; como respuesta, la CPU del Z80 entra en un estado 
especial de espera hasta que la sefíal WAIT se inactiva, momen- 
to en el que continúa la secuenciación normal. 

HALT (alto) es la sefial de reconocimiento que envía el Z80 
después de haber ejecutado una instrucción HALT. En este 
estado, el Z80 espera una interrupción externa mientras conti- 
nüa ejecutando ininterrumpidamente instrucciones NOP para 
refrescar la memoria. 

RESET es la señal que habitualmente inicializa el „P. Po- 
ne a “0” el contador del programa y los registros 1 y R; in- 
capacita el biestable de permisión de interrupciones y pone a 
“0” la función de interrupción. Normalmente se utiliza tras 
haber conectado la alimentación de la placa. 


CONTROL DE LA MEMORIA Y DE LA E/S 


El Z80 genera seis señales de control de la memoria y de los 
dispositivos de E/S. 

MREOQ es la señal de petición de memoria, que indica que 
la dirección presentada en el bus de direcciones es válida. A 
continuación puede efectuarse en la memoria una operación de 
lectura o escritura. 

М! es el ciclo de máquina 1 que corresponde a la fase 
tomar de una instrucción. 


IORQ es la petición de entrada/salida. Indica que la direc- 
ción de E/S presentada en los bits 0-7 del bus de direcciones es 
válida. A continuación puede efectuarse una operación E/S de 
lectura o escritura. IORQ se emite también junto con MI 
cuando el Z80 reconoce una interrupción; esta información 
puede ser aprovechada por microcircuitos externos para colocar 
el vector de respuesta a la interrupción en el bus de datos (las 
operaciones E/S normales nunca tienen lugar durante el estado 
М1; la combinación ТОКО más М! indica una situación de 
reconocimiento de interrupción). 

RD es la señal de lectura (utilizada en combinación con 
MREQ o IOREQ), e indica que el 780 está listo para leer el 
contenido del bus de datos en un registro interno. Puede utili- 
zarla un circuito externo de memoria o de E/S para depositar 
datos en el bus de datos. | 

WR es la señal de escritura utilizada en combinación con 
MREQ o IOREQ, e indica que el bus de datos contiene un 
dato válido, listo para ser escrito en el dispositivo especificado. 

RFSH es la señal de refresco. Cuando se activa, los siete bits 
inferiores del bus de direcciones contienen una dirección de re- 
fresco para memorias dinámicas. En ese momento se utiliza la 
señal, MREQ para proceder al refresco leyendo la memoria. 


Resumen del hardware 


Con esto terminamos la descripción de la organización in- 
terna del Z80. Los detalles exactos del soporte físico no son 
importanes en el contexto en que nos estamos moviendo, pero 
sí es necesario saber la función de cada uno de los registros, que 
debe conocerse perfectamente antes de pasar a los capítulos 
siguientes. Veremos a continuación el conjunto de instrucciones 
del Z80 y las técnicas básicas de programación de este micro- 
procesador. 


87 


Lud 
ще 
Ld 
к 
i. 
ще 
ж 
Jn 
ж. 
ж 
ж 
ж 
P 


d 
ше 
4 
ae 
Ld 
w 
als 
ж 
die 
Ed 
hd 
Ld 
E 


E d 
заби 
dg 


HII 


СЕЗНЕКЕ 


ШІП! 


Técnicas básicas 
de programación 


Introducción 


El objetivo de este capítulo es enseñar las técnicas elementa- 
les necesarias para escribir programas con el Z80. Presentare- 
mos aquí conceptos nuevos, como son los de organización de 
registros, bucles y subrutinas. Las técnicas de programación se 
centrarán exclusivamente en los recursos internos del 780. es 
decir, en los registros. Desarrollaremos, además, programas 
reales, en particular programas aritméticos que servirán para 
ilustrar los conceptos expuestos y para utilizar instrucciones 
reales. Veremos así qué instrucciones hay que emplear para 
manipular la información entre la memoria y el uP, y dentro 
de esta última. En el capítulo siguiente analizaremos con todo 
detalle las instrucciones de que dispone el 780. El capítulo 5 irá 
dedicado al estudio de las técnicas de direccionamiento, y el 6, 
al manejo de la información fuera del 780 o, lo que es lo 
mismo, a las técnicas de entrada y salida. 

En este capítulo aprenderemos, sobre todo, con la práctica. 
Al estudiar programas de complejidad creciente aprenderemos 
la función de las diferentes instrucciones y de los registros, y 
aplicaremos los conceptos ya desarrollados. No obstante, hay 
uno muy importante —-el de técnicas de direccionamiento— 
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оаа = 


Figura 3.1 


Registros del Z80. 


que, рог su aparente complejidad, no abordaremos hasta el 
capítulo 5. 

Tras esta breve introducción, pasaremos sin más a escribir 
algunos programas, empezando por los aritméticos. La figura 


3.1 recoge el “modelo para el programador” de los registros del 
780. 


GRUPO PRINCIPAL GRUPO ALTERNATIVO 


A 
registros 
generales 
SP 
(puntero de la pila) 
PC 
(contador del programa) 


REGISTROS 
DE INDICE 


Programas aritméticos 


Son programas aritméticos los de suma, resta, multiplicación 
y división. Los presentados aquí funcionarán con enteros, bien 
positivos binarios, bien en notación de complemento a dos, 
correspondiendo en este ültimo caso el bit de la izquierda al 
signo (véase en el capítulo 1 la descripción de la notación en 
complemento a dos). 


SUMA DE 8 BITS 


Vamos a sumar dos operandos de 8 bits denominados OPI 
y OP2, almacenados, respectivamente, en las direcciones de 
memoria ADR1 y ADR2. Llamaremos a la suma RES, y la 
almacenaremos en la dirección ADR3. Todo esto queda recogi- 
do en la figura 3.2. El programa encargado de realizar la suma 
es el siguiente: 


Figura 3.2 
Suma de 
ОР1 + OP2. 


8 bits; 


RES = 


Instrucciones Comentarios 


LD А, (АБВ!) CARGAR ОР! EN А 
LD НЕ, ADR? CARGAR LA DIRECCION DE OP2 
EN HL 
ADD A, (HL) SUMAR OP2 А ОР! 
LD (ADR3) A DEJAR EL RESULTADO RES EN 
ADR3 
MEMORIA 


(PRIMER OPERANDO) 


(SEGUNDO OPERANDO) 


(RESULTADO) 


DIRECCIONES 


Este es nuestro primer programa. Las instrucciones están a 
la izquierda, y los comentarios a las mismas, a la derecha. 
Examinémoslo con más detenimiento. Se trata de un programa 
de cuatro instrucciones (se llama instrucción a cada una de las 
líneas, y se ha expresado aquí de forma simbólica). El programa 
ensamblador traducirá cada una de esas Instrucciones a uno, 
dos, tres o cuatro bytes, aunque este extremo no es de nuestra 
incumbencia. 

En la primera línca se especifica que el contenido de ADRI 
se cargue en el acumulador A. Como se ve en la figura 3.2, ese 
contenido es el primer operando “OP1”; por tanto, el resultado 
de la primera instrucción es la transferencia de OP1 desde la 
memoria al acumulador; el movimiento se ilustra en la figura 
3.3. "ADRI" es una representación simbólica de la dirección 
real de 16 bits que se encuentra en la memoria. El símbolo 
ADR! se definirá en otra parte el programa; podría, por 
ejemplo, definirse como igual a la dirección “100”. 
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Figura 3.3 


LD А, (АОВ): 


de la memoria. 
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se carga OP1 


MEMORIA 


BUS DE DATOS 


BUS DE DIRECCIONES 


Esta instrucción de carga da lugar a una operación de 
lectura de la dirección 100 (véase la figura 3.3), cuyo contenido 
recorrerá el bus de datos hasta el acumulador, en el que queda 
depositado. Recordaremos del capítulo anterior que las opera- 
ciones lógicas y aritméticas actúan sobre el acumulador como 
uno de los operandos fuente (diríjase al mencionado capítulo 
para más detalles) Como queremos sumar los dos valores ОРІ 
y OP2, debemos empezar por cargar ОРІ en el acumulador; 
hecho esto, podremos sumar los contenidos del mismo, es decir, 
OP1 y OP2. El campo de la derecha de la instrucción se llama 
campo de comentario. El programa ensamblador lo ignora 
durante la traducción, pero de todas formas se incluye en el 
programa por razones de legibilidad. Para entender lo que hace 
el programa, es de capital importancia usar buenos comentarios 
(es lo que se llama documentar un programa). 

En este caso el comentario se explica a sí mismo; el valor de 
ОРІ, que se encuentra en la dirección ADRI, se carga en el 
acumulador A. 

El resultado de esta primera instrucción se ilustra en la 
figura 3.3. La segunda instrucción del programa es: 


LD HL, ADR2 


Obliga a “cargar ADR2 en los registros H y L”. Para leer 
en la memoria el segundo operando ОР2, antes debemos colocar 
su dirección en un par de registros del Z80, como H y L. A 
continuación podemos sumar el contenidio de la posición de 
memoria cuya dirección está en H y L al acumulador. 


ADD A, (HL) 


Como se ve en la figura 3.2, el contenido de la posición de 
memoria ADR2 es OP2, nuestro segundo operando. El conteni- 


do del acumulador es en este momento OPI, el primer operan- 
do. Como resultado de la ejecución de esta instrucción, se toma 
OP2 de la memoria y se suma а ОРІ, como ilustra la figura 3.4. 


280 MEMORIA 


ҰЯ 


2 < 


BUS DE DATOS 
A К?РРРРРРРРРРРА 


(—— OP2) 
2, 


ж Я 


(ADR2) 


Figura 3.4 
ADD A, (HL). BUS DE DIRECCIONES 


La suma se deposita en el acumulador. El lector recordará 
que, en el caso del Z80, el resultado de las operaciones aritméti- 
cas se deposita en el acumulador. En otros procesadores puede 
depositarse en otros registros o volver a la memoria. 

La suma de OP1 y OP2 está, pues, en el acumulador. Para 
completar el programa no tenemos más que transferir el conte- 
nido de aquél a la posición de memoria ADR3 para almacenar 
el resultado en la posición especificada. Esta operación es la que 
realiza la cuarta instrucción del programa: 


LD (ADR3), A 


Esta instrucción carga el contenido de А en la dirección 
ADR3. El efecto de la misma queda ilustrado en la figura 3.5. 


280 MEMORIA 


LLL 


BUS DE DATOS 


ADR3 ЖЖЖ” 


Figura 3.5 
LD (ADR3), A (guardar el acu- 
mulador en la memoria). BUS DE DIRECCIONES 


Antes de la ejecución de la operación ADD, el acumulador 
contiene OP1 (véase la figura 3.4). Tras la suma se ha escrito en. 
el mismo un nuevo resultado, que es “OP1 + OP2”. Recuérde- 
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se que el contenido de cualquier registro interno del microproce- 
sador, al igual que el de cualquier posición de memoria, per- 
manece invariable tras una operación de lectura. En otras 
palabras, leer el contenido de un registro o de una posición de 
la memoria no altera ese contenido. Este cambia únicamente en 
las operaciones de escritura. En el caso que nos ocupa, los 
contenidos de las posiciones de memoria ADR1 у ADR2 per- 
manecen invariables a lo largo de todo el programa. Sin 
embargo, tras la instrucción ADD, el contenido del acumulador 
si se modifica, porque se ha escrito en él la salida de la ALU; 
en consecuencia, su contenido anterior se pierde, 

En lugar de ADR1, ADR2 y ADR3 pueden utilizarse direc- 
ciones numéricas reales. Para trabajar con direcciones simbóli- 
cas es preciso utilizar lo que se llaman “seudoinstrucciones”, 
que especifican el valor de tales direcciones simbólicas para que 
el programa ensamblador pueda reemplazarlas por las direccio- 
nes fisicas verdaderas. Esas seudoinstrucciones podrían ser: 


ADRI = 100H 
ADR2 = 120H 
ADR3 = 200H 


Ejercicio 3.1: Consultando exclusivamente la tabla de instruccio- 
nes del final del libro, escríbase un programa que sume dos 
números almacenados en las posiciones de memoria ГОСТ y 
ГОС? y deposite el resultado en la posición LOC3. Compárese 
el programa con el que acaba de estudiarse. 


SUMA DE 16 BITS 


El programa de suma de 8 bits sirve únicamente para sumar 
números de 8 bits, es decir, números comprendidos entre 0 y 
255, si se trabaja en notación binaria absoluta. En la práctica, 
casi siempre es preciso trabajar con números de 16 bits o más 
y, por tanto, recurrir a la precisión múltiple. Presentaremos aquí 
algunos ejemplos de operaciones aritméticas en 16 bits que 
fácilmente podrán extrapolarse a 24, 32 o más (siempre múlti- 
plos de 8). Supondremos que el primer operando está almacena- 
do en las posicions de memoria ADR1 y ADR1-1. Como ОРІ 
es esta vez un número de 16 bits, necesitará dos posiciones de 
8 bits. De la misma forma, OP2 se almacena en ADR2 y 
ADR2-1. El resultado se deposita en las direcciones ADR3 
y ADR3-1. La figura 3.6 ilustra todo esto; H denota la mitad 
superior (bits 8 a 15) y L la inferior (bits O a 7). 


ro 


Figura 3.6 
Operandos de la suma de 16 
bits. 


MEMORIA 


ADR! —1 (OP1)H 
ADR1 (OP1)L 
ADR2-1 {ОР2)Н 
ADR2 (OP2)L 
ADR3-1 (RES)H 


Este programa sigue la misma lógica general que el anterior. 
En primer lugar, se suman las dos mitades inferiores de los dos 
operandos, porque el microprocesador sólo puede operar de 8 
en 8 bits. El acarreo que pudiera generar esa suma se almacena 
automáticamente en el bit interno de acarreo (“С”). А continua- 
ción se suman las dos mitades de orden superior de los dos 
operandos y el acarreo, y el resultado se guarda en la memoria. 
El programa es el siguiente: 


LD А, (ADRI) CARGAR ТА MITAD INFERIOR 
| . DE OPI 

LD HL, ADR2 CARGAR DIRECCION DE LA MI- 
TAD INFERIOR DE OP2 

ADD A, (НІ) | SUMAR ОР! Y OP2 (INFERIORES) 

LD (ADR3) А ALMACENAR EL RESULTADO (IN- 
FERIOR) 

LD А, (ADRI-1) CARGAR LA MITAD SUPERIOR 
DE ОР! 

DEC HL DIRECCION DE LA MITAD SUPE- 
RIOR DE OP2 

ADC A, (HL) (OP1 + OP2) SUPERIOR + ACA- 
RREO 

LD (ADR3-1) A ALMACENAR EL RESULTADO 

/ (SUPERIOR) 
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Las primeras cuatro instrucciones del programa son idénti- 
cas a las utilizadas para la suma de 8 bits de la sección anterior 
y dan lugar a la suma de las mitades menos significativas (bits 0 
а 7) de ОР! y OP2. La suma, llamada “RES”, se deposita en la 
posición de memoria ADR3 (véase la figura 3.6). 

El posible acarreo (“0” ó “1”) que pudiera resultar de la 
suma se almacena automáticamente en el bit de acarreo C del 
registro de estado (registro F). Si los dos números generan 
acarreo, el bit С será igual a “1”; en caso contrario, su valor 
será “0”. 

Las cuatro instrucciones siguientes son también básicamente 
iguales a las del programa de 8 bits a que nos estamos 
refiriendo. En este caso sirven para sumar las mitades más 
significativas (o mitades superiores, es decir, los bit 8 a 15) de 
OP1 y OP2 y cualquier acarreo, y almacenar el resultado en 
ADR3-1. 

Tras la ejecución de este programa de 8 instrucciones, el 
resultado de 16 bits aparece almacenado en las posiciones de 
memoria ADR3 y ADR3-1. Se observará, no obstante, que hay 
una diferencia entre las dos partes del programa; en efecto, la 
instrucción “ADD” utilizada en la primera parte (instrucción 
tercera) no aparece en la segunda. Dicha instrucción suma dos 
operandos, con independencia del acarreo. En la segunda parte 
se ha empleado en su lugar otra, llamada “ADC”, que suma 
dos operandos más cualquier acarreo que pudiera haberse 
producido y es, pues, necesaria para obtener un resultado 
correcto. Como la suma previamente ejecutada sobre las mita- 
des inferiores puede dar lugar a un acarreo, es preciso tener éste 
en cuenta en las instrucciones de la segunda parte. 

La cuestión que surge inmediatamente es: ¿qué ocurriría si 
la suma de las mitades superiores de los operandos también 
diese lugar a un acarreo? Hay dos posibilidades: la primera es 
suponer que se trata de un error; este programa está pensado 
para trabajar con resultados de hasta 16 bits, pero no de 17. La 
otra es incluir instrucciones adicionales para verificar precisa- 
mente la posibilidad de que se produzca un acarreo al final del 
programa. Se trata de la primera de una larga serie de decisio- 
nes que ha de tomar el programador. 

Nota: Hasta ahora hemos supuesto que la mitad superior de 
un operando se almacena “encima” de la inferior, es decir, en la 
dirección de memoria inmediatamente inferior. Pero las cosas 
no son necesariamente así, y, de hecho, en el 780 las direcciones 
se almacenan al revés: primero, la mitad inferior, y a continua- 
ción, la mitad superior en la siguiente posición de memoria. 
Con el fin de trabajar en una convención común para direccio- 
nes y datos, es recomendable que también éstos se almacenen 


Figura 3.7 
Almacenamiento de operandos 
en orden inverso. 


con la parte inferior sobre la superior. La situación se ilustra en 
la figura 3.7. 


MEMORIA 


ADR3 1 


Al trabajar con operandos de varios bytes, es importante 
tener en cuenta dos convenciones decisivas: 


— el orden en que se almacenan los datos en memoria, 
— la zona que señalan los apuntadores (byte inferior o byte 
superior). 


Los ejercicios 3.2 y 3.3 están pensados para aclarar estas 
cuestiones. 


Ejercicio 32: Vuelva a escribir el programa de suma de 16 bits 
con la organización de memoria descrita en la figura 3.7. 


Ejercicio 3.3: Supóngase ahora que ADRI no señala hacia la 
mitad inferior de ОРІ (como en las figuras 3.6 6 3.7), sino 
hacia la superior (figura 3.8). Vuélvase a escribir el programa 
teniendo en cuenta esta mueva convención. 


Es el programador quien decide cómo se almacenan los 
nümeros de 16 bits y también si las referencias de dirección 
señalan a la mitad inferior de dichos números o a la superior. 
Es otra decisión que hay que aprender a tomar durante el 
diseño de algoritmos y estructuras de datos. 


97 


MEMORIA 


ADR1-1 


---з- ADRI 


ADR2-1 (0P2)L 
—> ADR2 (OP2)H 


(RES)L 


Los programas que acabamos de examinar son programas 
tradicionales, que utilizan el acumulador. Veremos a continua- 
ción una alternativa al de 16 bits, que trabaja no con el 
acumulador, sino con algunas de las instrucciones especiales de 
16 bits de que dispone el Z80. Supondremos que los operandos 
están almacenados tal como describe la figura 3.6. El programa 
es el siguiente: 


LD НІ, (ADRI) CARGAR HL CON ОРІ 

LD ВС, (ADR2) CARGAR BC СОМ OP2 
ADD HL, BC SUMAR 16 BITS 

LD (ADR3) HL ALMACENAR RES EN ADR3 


Lo primero que llama la atención en este programa es que 
es mucho más corto que el anterior. Se dice que es más 
“elegante”. Con ciertas limitaciones, los registros H y L del Z80 
pueden utilizarse como un acumulador de 16 bits, 


ADR3-1 
—>  ADR3 


Figura 3.8 
Punteros del byte superior. 


Ejercicio 3.4: Con las instrucciones de 16 bits que acabamos de 
presentar, escribase un programa de suma para operandos 
de 32 bits, suponiendo que éstos se almacenan como describe la 
figura 3.9. 


Respuesta: 


LD HL, (АБВ!) 
LD ВС, (ADR2) 
ADD HL, BC 


| 
| 
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Figura 3.9 
Suma de 32 bits. 


LD (ADR3, HL 
LD HL, (ADRI + 2) 
LD ВС, (ADR? + 2) 
ADC HL, BC 

LD (АОВЗ + 2), HL 


MEMORIA 
ADRI + 3 SUPERIOR 
OPR1 


Арві INFERIOR 
SUPERIOR 

OPR2 
ADR2 INFERIOR 


SUPERIOR 
RES 


ADR3 INFERIOR 


Ahora que ya sabemos programar la suma binaria, podemos 
pasar a la resta. E 


RESTA DE NUMEROS DE 16 BITS 


La resta de 8 bits es demasiado sencilla, así que la dejare- 
mos como ejercicio y pasaremos directamente al problema de 
restar nümeros de 16 bits. Como es habitual, los dos nümeros 
OPI y OP2 se almacenan en las direcciones ADRI y ADR2, 
suponiendo una disposición de memoria similar a la ilustrada 
en la figura 3.7. Para restar se sustituye la instrucción ADD por 
la SBC. 


Ejercicio 3.5: Escribir un programa de resta. 


Figura 3.10 
Carga de 16 bits: 
LD HL, (ADR1). 


100 


El programa aparece a continuación, y las rutas seguidas 
рог los datos se muestran en la figura 3.10. 


LD НІ, (АБВ ОРІ EN HL 

LD DE, (АПЕ2) OP2 EN DE 

AND A ELIMINAR ACARREO 
SBC HL, DE ОРІ — ОР? 

LD (ADR3) HL RES ЕМ ADR3 


El programa es básicamente igual al de suma de 16 bits. Sin 
embargo, mientras que el 780 tiene dos tipos de suma en 
registros dobles —ADD y ADC—, sólo cuenta con una resta: 
SBC. Como consecuencia de ello, ha habido que introducir dos 
cambios. 


MEMORIA 


El primero es el uso de SBC en lugar de ADD. 

El segundo es la instrucción “AND A”, utilizada para 
eliminar la bandera de acarreo antes de restar. Esta instrucción 
no modifica el valor de A. 

La precaución es necesaria, porque el Z80 dispone de dos 
formas de suma, con y sin acarreo, еп los registros Н y Г, pero 
sólo de una resta, SBC, o resta con acarreo, cuando trabaja en 
el par de registro HL. Como SBC tiene en cuenta automática- 
mente el valor del bit de acarreo, es preciso poner éste a 0 antes 


de ejecutar la operación, y eso es precisamente lo que hace la 
instrucción “AND A”. 


H 1 


ADRI 


ADRI +1 


Ejercicio 3.6: Escríbase de nuevo el programa de resta sin usar las 
instrucciones especiales de 16 bits. 


Ejercicio 3.7: Escríbase el programa de resta para operandos de 8 
bits. 


Hay que recordar que, en la aritmética en complemento a 
dos, el valor final de la bandera de acarreo no tiene significado. 
Si, como resultado de la resta, se produce una situación de 
desbordamiento, entrará en juego el bit correspondiente (bit V) 
del registro de estado, cuyo valor podrá verificarse. 

Los ejemplos que acabamos de estudiar corresponden a 
sencillas sumas y restas binarias, pero es necesario trabajar en 
otras formas de representación aritmética, y en particular en 
BDC. 


Aritmética BCD 


SUMA BCD DE 8 BITS 


El concepto de notación aritmética BCD se expuso en el 
capítulo 1, pero recordaremos aquí sus características esenciales. 
Se utiliza, sobre todo, en aplicacions contables, en las que es de 
rigor conservar en el resultado todas las cifras significativas. En 
notación BCD se emplea un nibble de 4 bits para almacenar 
una cifra decimal (de O a 9), de manera que un byte de 8 bits 
representa dos cifras BCD (BCD condensado). Veamos ahora 
cómo se suman dos bytes de dos cifras BCD cada uno. 

Para identificar la naturaleza del problema, analizaremos 
antes algunos ejemplos numéricos. 

Sea la suma de “01” y “02”: 


“01” se representa como 0000 0001 
“02” se representa como 0000 0010 
El resultado es 0000 0011 


que es la representación BCD de “03” (si duda al deducir los 
equivalentes BCD, consulte la tabla de conversión del final del 
libro). Este caso ha sido muy fácil. Veamos otro: 


“08” se representa como 0000 1000 
“03” se representa como 0000 0011 


Ejercicio 3.8: Calcúlese la suma de los dos números de arriba en 
notación BCD. ¿Qué se obtiene como resultado? 


Respuesta: Si el resultado es “0000 1011”, lo que ha obtenido es 


la suma binaria de 8 y 3, es decir, 11 en representación 
binaria. Lo que ocurre es que “1011” no es un código BCD 
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válido. De lo que se trata es de obtener la representación 
BCD de “11”, es decir, 00010001. 


El problema radica en que la representación BCD utiliza 
únicamente las primeras diez combinaciones de cuatro cifras 
para codificar los símbolos decimales O a 9. Las seis posibles 
combinaciones que quedan no se usan, y “1011” es una de ellas, 
En otras palabras, siempre que la suma de dos cifras BCD sea 
mayor que 9, hay que añadir 6 al resultado para saltar por 
encima los seis códigos que no se usan. 

En efecto, al sumar la representación binaria de “6” a 1001: 


1011 (resultado binario no permitido) 
+ 0110 (+6) 


se obtiene: 00010001 


que es “11” en notación BCD. Por fin hemos obtenido el 
resultado correcto. 

Este ejemplo ilustra una de las dificultades básicas que 
plantea la notación BCD, a saber: la necesidad de compensar la 
existencia de seis códigos que no se usan. Para corregir el 
resultado de la suma binaria se utiliza una instrucción “DAA”, 
llamada “ajuste decimal” (la instrucción suma 6 я el resultado 
es mayor que 9). 

El mismo ejemplo servirá para ilustrar el problema siguien- 
te. El acarreo procede de la cifra BCD inferior (la de la derecha) 
y pasa a la de la izquierda. Es preciso tener en cuenta este 
acarreo interno y sumarlo a la segunda cifra BCD, cosa que 
hace automáticamente la instrucción de suma. Sin embargo, con 
frecuencia conviene detectar este acarreo interno del bit 3 al 4 
(“semiacarreo”), para lo que se emplea la bandera H. 

Ilustraremos todo esto con el siguiente ejemplo, un progra- 
ma que suma los números BCD “11” y “22”: 


LD А, ИН CARGAR EL LITERAL BCD “11” 

ADD A, 22H SUMAR EL LITERAL BCD “22” 

DAA AJUSTE DECIMAL DEL RESULTA- 
DO 


LD (ADR) A ALMACENAR EL RESULTADO 


En este programa hemos introducido un símbolo nuevo 
——“H”— que, situado dentro del campo del operando de la 
instrucción, indica que el dato al que sigue se expresa en 
notación hexadecimal. Las representaciones hexadecimal y BCD 
de las cifras “0” a “9” son idénticas. En este caso queremos 
sumar los literales (o constantes) “11” y “22”, y almacenar el 
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resultado en la dirección ADR. Cuando el operando se especifi- 
ca como parte de una instrucción, como el ejemplo que nos 
ocupa, se habla de direccionamiento inmediato (en el capitulo 5 
se analizarán en detalle las diversas formas de direccionamien- 
to). Almacenar el resultado en una dirección especificada, como 
LD (ADR), A se llama direccionamiento absoluto cuando ADR 
representa una dirección de 16 bits. 


MEMORIA 


(ADR) 


(RESULTADO) 


Este programa es análogo al de suma binaria de 8 bits, pero 
con la nueva instrucción “DAA”, de la que mostraremos el 
funcionamiento con un ejemplo. Empecemos por sumar “11” y 
“22” en BCD: 


Figura 3.11 
Almacenamiento de cifras BCD. 


00010001 (11) 
+ 00100010 (22) 
= 00110011 (33) 


33 


Е resultado, alcanzado con las reglas de la suma binaria, 
es correcto. 

Sumemos ahora “22” y “39” con las mismas reglas de la 
suma binaria: 


00100010 (22) 
+ 00111001 (39) 


= 01011011 
ed 
5 7? 


: “1011” es un código BCD no permitido, porque en BCD se 
utilizan sólo los primeros diez códigos binarios y se saltan los 
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seis siguientes; por tanto, habrá que hacer aquí lo mismo, es 
decir, sumar 6 al resultado: 


01011011 (resultado binario) 
+ 0110 (6) 


= 01100001 (61) 
6 i 


que es el resultado BCD correcto. 


Ejercicio 3.9: ;Podría colocarse en el programa la instrucción 
DAA después de la LD ( ADR), A? 


RESTA BCD А 

А primera vista, la resta еп BCD parece una cosa muy 
complicada. La operación se realiza sumando el complemento a 
diez del número, igual que se sumaba el complemento a dos para 
realizar la resta binaria. El complemento a 10 se calcula deter- 
minando el de 9 y sumando 1, lo que en un microprocesador 
normal consume de tres a cuatro operaciones; sin embargo, el 
780 dispone de una potente instrucción DAA, que simplfica las 
cosas. 

La instrucción DAA ajusta automáticamente el valor del 
resultado del acumulador en función del valor de las banderas 
С, Ну М, antes de DAA, a su valor correcto (véase el capítulo 
siguiente para más detalles sobre ОАА). 


SUMA BCD DE 16 BITS 


Esta operación se realiza exactamente igual que la binaria 
correspondiente. El programa es el siguiente: 


LD A,(ADRI) CARGAR (ОРІ) EN A 
LD  HÁHL,(ADR2% CARGAR ADR2 EN HL 


ADD A,(HL) (OP1 + OP2) INFERIOR 

DAA AJUSTE DECIMAL” + 

LD  (ADR3,A ALMACENAR EL RESULTADO (IN- 
FERIOR) 

LD А, (ADR1+1) CARGAR (OP1)H EN А 

INC HL PUNTERO А АОК? + 1 

ADC A, HL) (ОРІ + OP2) SUPERIOR + ACA- 
RREO 


jE UE 


DAA AJUSTE DECIMAL 
LD (АОКЗ + 1), A ALMACENAR EL RESULTADO 
(SUPERIOR) 


RESTA EN BCD EMPAQUETADO 


Ya se han descrito los procedimientos elementales de suma y 
resta BCD. Sin embargo, en la práctica habitual, los números 
BCD tienen un número variable de bytes. Como ejemplo 
simplificado de resta en BCD empaquetado supondremos que 
las dos cantidades localizadas en N1 y N2 tienen el mismo 
número de bytes BCD, número que se llama CUENTA. La 
organización de registros y memoria se muestra en la figura 
3.12. El programa es el siguiente: 


BCDPAK LD B, CUENTA 
LD DE, N2 
LD HL, Ni 
AND А ELIMINAR ACARREO 
MENOS LD A, (DE) BYTE № 
SBC А, (НІ)  N2- NI 


DAA 

LD (HL), A ALMACENAR EL RE- 
: SULTADO 

INC DE 

INC HL 


DJNZ MENOS DECREMENTAR B Y 
REPETIR HASTA B=0 


N1 y N2 son las direcciones en que están almacenados los 
números BCD, direcciones que se cargan en los pares de 
registros DE y HL: 


BCDPAK LD B, CUENTA 


LD - DE, N2 
LD HL, Ni 


A continuación, antes de la prifnera resta, hay que eliminar 
el bit de acarreo. Ya se ha dicho que hay varias formas de 
hacerlo. Una de ellas es: 


AND A 
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B CUENTA 


Figura 3.12 
Resta en BCD empaquetado 
N1 N2 - Nf. е 


El primer byte de N2 se carga en el acumulador y se le resta 
el primero de N1. A continuación se utiliza la instrucción DAA 
para obtener el valor BCD correcto: 


MENOS LD A, (DE) 
SBC А, (HL) 
DAA 


El resultado se almacena еп МІ: 


LD (HL) A 

Por último, se incrementan los punteros de los bytes en 
curso: 

INC DE 

INC HL 


Se decrementa el contador y se ejecuta el bucle de resta hasta 
que alcance el valor “0”: 


DJNZ MENOS 
La instrucción DJNZ es una instrucción especial del Z80 que 


decrementa el registro B y salta —si no es 0— en una sola 
operación. 
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Multiplicación 


Ejercicio 3.10: Compárese el programa que acaba de presentarse 
con el de suma binaria de 16 bits. ¿Dónde está la diferencia? 


Ejercicio 3.11: ¿Son intercambiables los papeles de DE y HL? 
(Un consejo: cuidado con SBC.) 


Ejercicio 3.12: Escríbase un programa de resta en BCD de 16 
bits. 


BANDERAS BCD 


En notación BCD, la bandera de acarreo que aparece como 
resultado de una suma indica que el resultado es superior a 99. 
La situación es diferente a la que se daba en complemento a dos, 
porque las cifras BCD están representadas en auténtica nota- 
ción binaria; por el contrario, la presencia de bandera de 
acarreo tras una resta indica un defecto. 


TIPOS DE INSTRUCCIONES 


Hemos utilizado ya dos tipos de instrucciones del micropro- 
cesador: instrucciones LD, que cargan el acumulador a partir 
de direcciones de memoria o almacenan su contenido en direc- 
ciones especificadas; se llaman instrucciones de transferencia de 
datos. 

Instrucciones aritméticas, como ADD, SUB, ADC y SBC, 
que ejecutan operaciones de suma y resta. Pronto veremos en 
este mismo capítulo otras instrucciones de la ALU. 

Pero hay otros tipos que todavía no hemos usado: se trata 
de las instrucciones de salto, que modifican el orden de ejecu- 
ción del programa; recurriremos a esta clase de instrucciones en 
el próximo ejemplo. Conviene observar que a las instrucciones 
de salto se les llama también de bifurcación en situaciones 
condicionales, es decir, en puntos del programa en los que se 
toma una decisión lógica. Como sugiere el nombre de bifurca- 
ción, la presencia de esa instrucción escinde el camino único del 
programa en dos divergentes. 


Analicemos a continuación un problema aritmético más 
complicado: la multiplicación de números binarios. Antes de 
redactar el algoritmo de la operación, examinaremos una multi- 
plicación decimal corriente; sea la de 12 por 23: 
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iy 12 (multiplicando = MPD) 
` мавт х 23 (multiplicador = МРК) 
36 (producto parcial) 
+24 


= 276 (resultado final) 


La operación se lleva a cabo multiplicando la cifra de la 
derecha del multiplicador por el multiplicando, es decir: “3” 
x “12”; el producto parcial es “36”; a continuación se multipli- 
ca la siguiente cifra del multiplicador, “2”, por “12”, y el 
resultado, “24”, se suma al producto parcial. 

Pero todavía falta una operación: 24 se escribe desplazado 
una posición hacia la izquierda (decimos que 24 se desplaza a la 
izquierda una posición, pero igual podriamos haber dicho que el 
producto parcial 36 se desplaza una posición a la derecha). 

Los dos números, correctamente desplazados, se suman, y 


.а51 se obtiene el producto 276. Es algo muy sencillo, y las cosas 


ocurren exactamente de la misma forma en notación binaria. 
Veamos, por ejemplo, la multiplicación de 5 por 3: 


(5) 101 (multiplicando 
(3) х 011 (multiplicador) 
101 (producto parcial) 
101 
000 


(15) 01111 (resultado final) 


Realizaremos la multiplicación exactamente igual que en el 
ejemplo que acabamos de ver. La representación formal del 
algoritmo aparece en la figura 3.13. Se trata de un diagrama de 
flujo —el primero del libro—, y lo analizaremos con cierto 
detalle. 

El diagrama de flujo es una representación simbólica del 
algoritmo que acabamos de seguir. Cada rectángulo representa 
una orden que debe ejecutafse, y que habrá que transformar en 
una o más instrucciones del programa. En cada uno de los 
símbolos romboidales hay que llevar a cabo una comprobación, 
ya que son puntos de bifurcación del programa. Si el resultado 
de la comprobación es afirmativo, la bifurcación conduce el 
flujo del programa a una posición determinada; en caso negati- 
vo, lo conduce a una posición diferente. La idea de bifurcación 
se expondrá más adelante, en el propio programa. El lector 
deberá, por el momento, estudiar atentamente el diagrama de 
flujo hasta que adquiera la completa seguridad de que represen- 


ta efectivamente el algoritmo de multiplicación. Obsérvese que 
del rombo inferior parte una flecha que se dirige al superior; se 
debe a que esa porción del diagrama debe ejecutarse ocho 
veces, una por cada bit del multiplicador. Esta disposición del 
programa que provoca el reinicio de una misma operación en 
un mismo punto es lo que se llama un bucle. 


PONER RESULTADO A CERO) 


N 
¿BmS (MPR) = 1? 9 
RESULTADO = 
RESULTADO + MPD 


DESPLAZAMIENTO IZQ (1) 
MPD, О DESPLAZAMIENTO, 
DER (1) RES 


Figura 3.13 < 
Diagrama de flujo del algoritmo 


básico de multiplicación. TERMINADO 


Ejercicio 3.13: Ejecútese la multiplicación binaria de “4” por “7” 
utilizando el diagrama de flujo y verificando el resultado, que 
debe ser “28”. Pruébese de nuevo si el que se obtiene es otro, 
porque sólo quien sea capaz de seguir el diagrama hasta dar 
con el valor correcto estará en condiciones de transformarlo en 
un programa, 


MULTIPLICACION 8 POR 8 


Vamos, por fin, a transformar el diagrama de flujo en un 
programa para el Z80, programa que aparece completo en la 
figura 3.14 y que estudiaremos en detalle. Como se recordará 
del capítulo 1, programar consiste en este caso en “traducir” el 
diagrama de flujo de la figura 3.13 al programa de la 3.14. Cada 
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Figura 3.14 
Programa de multiplicación 
8 x 8. 
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uno de los recuadros del diagrama dará lugar a una o más 
instrucciones, 

Se supone que MPR y MPD han recibido ya un valor 
concreto. 


MPY88 LD ВС, (MPRAD) CARGAR MULTIPLI- 
CADOR EN С 
LD В, 8 В ES EL CONTADOR 
DE ВТ 
LD DE, (MPDAD) CARGAR EL MULTI- 
PLICANDO EN E 


LD D,0 BORRAR D 
LD HL, 0 PONER EL RESULTA- 
DO A 0 
MULT SRL С DESPLAZAR EL BIT 


DEL MULTIPLICA- 
DOR AL ACARREO 
JR NC, NOADD VERIFICAR EL 


ACARREO 

ADD HL, DE SUMAR MPD AL RE- 
SULTADO 

NOADD SLA E DESPLAZAR MPD A 

LA IZQUIERDA 

RL D LLEVAR BIT A D 

DEC B DECREMENTAR 
CONTADOR DE DES- 
PLAZAMIENTO 

JP NZ, MULT REPETIR SI 


CONTADOR 4 0 
LD (RESAD), HL ALMACENAR EL RE- 
SULTADO 


La primera casilla del diagrama es la de inicialización, necesa- 
ria porque antes de nada hay que poner а 0 una serie de registros 
о posiciones de memoria para que el programa trabaje en ellos. 
Los registros que se utilizarán en el programa de multiplicación 
aparecen en la figura 3,15. 

Se utilizan tres pares de registros del Z80. El multiplicador 
de 8 bits se supone que reside en la posición de memoria, 
МРКАО; el multiplicando, MPD, ocupa la dirección MPDAD, 
y los dos se cargarán еп los registros Су E, respectivamente 
(véase la figura 3.15). El registro B trabaja como contador. 

Los registros D y E albergan el multiplicando a medida que 
se desplaza de bit en bit hacia la izquierda. 


Figura 3.15 
Registros usados en la multipli- 
cación 8 x 8. 


(CONTADOR) 


al 
<< 2224 
МЭМ 


(MPRAD) 


1% 


mh xj 


Obsérvese que, aunque en un principio basta con cargar C y 
E, hay que prever 16 bits para que también puedan cargarse B 
y D a partir de la memoria; se ponen, respectivamente, a "8" y 
“0”, 

Por último, hay que tener en cuenta que el resultado de una 
multiplicación de 8 por 8 bits puede ocupar hasta 16 bits, 
porque 28 x 28 = 216: por tanto, hay que reservar para el 
resultado dos registros, que son los H y L, como indica la 
figura 3.15. 

El primer paso es cargar los registros B, C y E con los 
contenidos adecuados e iniciar el resultado (el producto parcial) 
al valor “0”, tal como especifica el diagrama de flujo de la figura 
3.13. Todo ello se consigue con las siguientes instrucciones: 


ZZ 
РЕД 
272 
AENA 


RES 


(MPDAD) 


(RESAD) 


MPY88 LD BC,(MPRAD) 
LD В, 8 
LD DE,(MPDAD) 
ED ро. 
LD HL,O 


Las tres primeras instrucciones cargan MPR en el par de 
registros BC, el valor “8” en el registro B y MPD en el par de 
registros DE, respectivamente. Como MPR y MPD son pala- 
bras de 8 bits, se cargan, de hecho, en los registros C y E, 
mientras que las palabras de la memoria que les siguen pasan a 
B y D. La situación se ilustra en las figuras 3.16 y 3. 17. La 
siguiente instrucción pone a O el contenido de D. 
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Figura 3.16 


LD BC, (MPRAD). 


Figura 3.17 


LD DE, (MPDAD). 
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En este programa de multiplicación, el multiplicando se 
desplaza hacia la izquierda antes de sumarlo al resultado 
(recuérdese que también se puede desplazar el resultado a la 
derecha, como indica la cuarta casilla del diagrama de flujo de 
la figura 3.13). A cada paso, el multiplicando MPD se desplaza 
hacia-el registro D, que, por tanto, debe iniciarse al valor “0”, 
operación que lleva a cabo la instrucción cuarta. La quinta 
pone a O de una vez los contenidos de los registros H y L. 


MEMORIA 


MEMORIA 


E 
E 
— | 


El siguiente paso del diagrama de flujo consiste en poner a 
prueba el bit menos significativo (el de la derecha) del multipli- 
cador, MPR. Si resulta ser “1”, el valor de MPD se añade al 
resultado parcial; en caso contrario, no se añade. Para ello 
hacen falta tres instrucciones: 


MULT SRL C | 
JR NC, NOADD | 
ADD HL, DE 


El primer problema a resolver es la verificación del bit 
- menos significativo del multiplicador contenido en el registro C. 

Podemos usar para ello la instrucción BIT del 780, que permite 
comprobar cualquier bit de cualquier registro, pero en este caso 
lo que nos interesa es crear un programa con un bucle lo más 
sencillo posible. Para utilizar la instrucción BIT, tendríamos 
que comprobar, primero, el bit 0; luego, el 1, y asi sucesiva- 
mente hasta llegar al 7, lo que exigiría una instrucción diferente 
cada vez, algo incompatible con un solo bucle. Para acortar el 
programa hay que buscar otro camino, y en este caso hemos 
decidido trabajar con una instrucción de desplazamiento. 

Nota: Hay una forma de utilizar la instrucción BIT y un 
bucle, pero exigiría que el programa se modificase a sí mismo, 
una práctica que, por el momento, evitaremos. 

SRL es un nuevo tipo de operación que se ejecuta dentro de 
la unidad aritmética y lógica. Significa desplazamiento lógico a la 
derecha. Tras un desplazamiento lógico a la derecha, aparece un 
“0” en la posición del bit 7; por el contrario, tras un desplaza- 
miento aritmético a la derecha, el bit que ocupa la posición 7 
adopta el mismo valor que antes tenía dicha posición. En el 
próximo capítulo se describirán las diversas operaciones de 
desplazamiento. El resultado de la instrucción SRL С viene 
mostrado en la figura 3.15 por una flecha que sale del registro 
C y se dirige hacia el cuadrado que designa el bit de acarreo 
(“С”). En este punto, el bit de la derecha del MPR estará en el 
bit de acarreo C, el que se comprueba. 

La instrucción siguiente, “JR NC, NOADD”, es una opera- 
ción de salto que significa: “si no hay acarreo” (NC), saltar a la 
dirección NOADD (etiqueta). Si el contenido del bit de acarreo 
es “0” (no hay acarreo), el programa salta a la dirección 
NOADD; si el contenido de C es “1” (hay acarreo), no se 
produce bifurcación, y se ejecuta la siguiente instrucción de la 
secuencia, en este caso “ADD HL, DE”. 

Esta instrucción dice que hay que sumar los contenidos de 
D y E a los de Ну L, y dejar el resultado еп Н y L. Como E 
contiene el multiplicando, MPD (véase la figura 3.15), resulta 
que la instrucción suma dicho multiplicando al resultado par- 
cial. 

En este punto, con independencia de que MPD se haya 
sumado o no al resultado, hay que desplazar el multiplicando a 
la izquierda (cuarto recuadro del diagrama de flujo de la figura 
3.13). Para ello se usa la instrucción: 


NOADD SLA E 


SLA significa “desplazamiento aritmético a la izquierda”. Como 
ya hemos explicado, hay dos tipos de operaciones de desplaza- 
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miento: lógico y aritmético. Este es el aritmético. En el caso de 
desplazamiento a la izquierda, SLA especifica que el bit de la 
parte derecha del registro (el menos significativo) sea “0”, como 
en el caso de SRL que vimos antes. 

Supongamos, por ejemplo, que el contenido inicial del regis- 
tro E fuera 00001001. Tras la instrucción SLA, ese contenido 
será 00010010, y el bit de acarreo valdrá 0. 

Pero, como se ve en la figura 3.15, lo que nos interesa es 
desplazar el bit más significativo (BMS) de E directamente a D 
(este movimiento viene ilustrado por la flecha que va de E a 
D); sin embargo, no hay ninguna instrucción para desplazar un 
doble registro, como el D y E, de una vez. Una vez desplazados 
los contenidos de D y E, el bit de la izquierda habrá “caido” en 
el bit de acarreo; por tanto, hay que recoger ese bit y despla- 
zarlo al registro D, y para ello sirve la instrucción siguiente: 


RL D 


RL es también una operación de desplazamiento, pero de 
distinto tipo. Significa “rotación circular a la izquierda”. En una 
operación de rotación circular, al contrario que en una de 
desplazamiento, el bit que llega al registro es el contenido del bit 
de acarreo С (véase la figura 3.18), que es justamente lo que nos 
interesa. El contenido de C se carga en el extremo derecho de 
D, lo que, de hecho, equivale a transferir el bit izquierdo de E. 

Esta secuencia de dos instrucciones se muestra en la figura 
3.19. Como se ve, el bit identificado por X en la posición más 
significativa de Е pasa primero al bit de acarreo у a continua- 
ción a la posición menos significativa de D. 

En este punto, como indica el diagrama de flujo de la figura 
3.13, hay que señalar el siguiente bit de MPR y comprobar si es 
el octavo. Esto se consigue decrementando el contador de bits 
del registro В (figura 3.15). De decrementar el registro, se 
encarga la instrucción: 


DEC B 


que es una instrucción de decremento de resultado evidente. 

Por último, es preciso comprobar si el contador ha dismi- 
nuido hasta 0, lo que se consigue examinado el valor del bit Z. 
Como recordará el lector, la bandera Z (0) indica si la opera- 
ción aritmética previa (DEC, por ejemplo) ha producido un 
resultado nulo. Obsérvese, sin embargo, que DEC HL, DEC 
BC, DEC DE, DEC IX y DEC SP no afectan a la bandera Z 
mencionada. Si el contador no es "0", quiere decir que la 
operación no ha terminado, y que hay que ejecutar una vez más 


Figura 3.18 
Desplazamiento y rotación 
circular. 


Figura 3.19 
Desplazamiento de E a D. 


el bucle del programa, de lo que se encarga la instrucción 
siguiente: 


JP NZ, MULT 


DESPLAZAMIENTO A LA IZQUIERDA 
ACARREO 


ROTACION A LA IZQUIERDA 


E E 
ES 


Se trata de una instrucción de salto, la cual especifica que 
siempre que el bit Z no sea 0 (NZ significa no cero), hay que 
saltar a la posición MULT. De esta forma se cierra el bucle del 
programa, que se ejecutará una y otra vez hasta que el valor de 
B se reduzca a 0. En ese momento, el bit Z adquirirá un valor 
no nulo, y la instrucción JP NZ dejará de actuar, lo que dará 
lugar a que se ejecute la instrucción siguiente de la secuencia, a 
saber: 


LD (RESAD), HL 


Esta instrucción almacena el contenido de H y L, es decir, el 
resultado de la multiplicación, en la dirección RESAD. Obsér- 
vese que la instrucción transfiere los contenidos de ambos 
registros a dos posiciones de memoria consecutivas: RESAD y 
RESAD + 1. Almacena 16 bits de una vez. 
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Ejercicio 3.14: ¿Seria capaz de escribir el programa de multiplica- 
ción que acabamos de estudiar sustituyendo la instrucción 
SRL C por la BIT (descrita en el capítulo siguiente)? ¿Qué 
inconveniente tiene? 


Tratemos ahora de mejorar el programa, si tal cosa es 
posible. 


Ejercicio 3.15: ¿Puede sustituirse JR por JP al final del progra- 
ma? En caso afirmativo, ¿qué ventaja tiene el cambio? 


Ejercicio 3.16: ¿Puede utilizarse DJNZ para acortar el final del 
programa? 


Ejercicio 3.17: Estúdiense las instrucciones LD D,0 y LD НІ,0 
del principio del programa; ¿pueden sustituirse por 


XOR A 

LD D,A 
LD ҢА 
LD L,A 


En caso afirmativo, ¿qué efecto ejercerían sobre el tamaño 
(número de bytes) y la velocidad? 


En la mayor parte de los casos, el programa que acabamos 
de desarrollar será un subrutina que tendrá como instrucción 
final RET (return, vuelta). El mecanismo de subrutina se expli- 
cará más adelante en este mismo capítulo. 


UN EJERCICIO IMPORTANTE 


El que acabamos de ver es el primer programa realmente 
importante del libro. Incluye instrucciones muy diversas: de 
transferencia (LD), aritméticas (ADD), lógicas (SRL, SLA, RL) y 
de salto (JR, JP) y cuenta, además, con un bucle de siete 
instrucciones que empiezan en la dirección MULT y se ejecutan 
varias veces seguidas. Para aprender a programar es imprescindi- 
ble entender perfectamente este programa. Es más largo que los 
sencillos programas aritméticos de los capítulos anteriores, y 
resulta imprescindible examinarlo con detenimiento. А conti- 
nuación se propone un ejercicio de la mayor importancia, y es 
decisivo que el lector lo realice por completo y correctamente 
antes de seguir, porque será la única demostración de que ha 
asimilado los conceptos expuestos con anterioridad. Quien lo 
resuelva correctamente tendrá la seguridad de haber comprendi- 


do cabalmente la forma en que el microprocesador manipula la 


т información, la desplaza entre los registros y la memoria, y Ig ` 


procesa. Quien no haga el ejercicio o quien no lo resuelva 
correctamente es muy probable que tropiece con dificultades al 
escribir sus propios programas. Aprender a programar exige 
práctica; por tanto, tome un papel, o utilice la figura 3.20, y 
haga el 


Ejercicio 3.18: Siempre que se escribe un programa es preciso 
comprobarlo a mano, para tener la seguridad de que propor- 
ciona resultados correctos, y eso es justamente lo que vamos a 
hacer ahora; la finalidad de este ejercicio es rellenar la tabla 
de la figura 3.20. 


ETIQUETA INSTRUCCION ЕЕ 
(АСАВВЕО) 


Figura 3.20 


ре кы г шш 1 
cación. 


La respuesta puede escribirse directamente en la figura o en 
un papel aparte. De lo que se trata es de indicar el contenido de 
cada uno de los registros que intervienen en el programa tras la 
ejecución de cada una de las instrucciones. En la figura 3.20 
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Figura 3.21 
El programa de multiplicación 
tras una instrucción. 


Figura 3.22 PE 
El programa de multiplicación 
tras dos instrucciones. 
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aparecen todos los que forman parte del programa de la figura 
3.14, que son, de izquierda a derecha: B y C, el acarreo C, D y 
E, y H y L. En la parte izquierda de la figura se anotan la 
etiqueta, si existe, y la instrucción que va a ejecutarse. En la 
parte derecha se anota el contenido de cada uno de los registros 
tras la ejecución de la instrucción que figura a la izquierda. Si el 
contenido de un registro no se conoce (es indefinido), se sefiala 
tal circunstancia con un trazo. Como ejemplo, empezaremos a 
rellenar las primeras filas de la tabla: 


ETIQUETA |INSTRUCCION вс | 
MPY88 | LD ВС, (0200) 03 


Suponemos en este caso que estamos multiplicando “3” 
(MPR) por “5” (MPD). 

La primera instrucción que se ejecuta es “LD BC, 
(MPRAD)". El contenido de la posición de memoria MPRAD se 
carga en los registros B y C. Hemos supuesto que MPR es 
igual a 3, es decir, “00000011”. Tras la ejecución de esta 
instrucción, el contenido del registro C pasa a ser "3". Obsérve- 
se que la instrucción también carga B con lo que siga a MPR 
en la memoria. La instrucción siguiente se ocupa de ello y carga 
en B el valor “8”, como ilustra la figura 3.22. Por el momento, 
los contenidos de D y E y H y L están indefinidos. La 
instrucción LD no perturba al bit de acarreo, de tal manera 
que también está indefinido el contenido del mismo. 


LD BC, (0200) 
LD B, 08 


La situación tras la ejecución de las primeras cinco instruc- 
ciones del programa (justo antes de MULT) se ilustra en la 
figura 3.23. 

La instrucción SRL lleva a cabo un desplazamiento lógico a 
la derecha, de manera que el bit de ese extremo de MPR pasa 
al bit de acarreo. Como se ve en la figura 3.24, el contenido de 
MPR tras el desplazamiento es “0000 0001”. El bit de acarreo 
C vale ahora “1”. La operación no ha afectado a los demás 


registros. Ahora, debe continuar usted mismo rellenando la 
tabla. 

Al final de este capítulo, en la figura 3.42, se recoge una 
segunda repetición. 


ШТІ ТІГЕ ЕЗЕЗЕЗЕНГІНЕ 


LD BC, (0200) 
LD B, 08 
LD DE, (0202) 


Figura 3.23 LD D, 00 


El programa de multiplicación 
tras cinco instrucciones. 


ETIQUETA INSTRUCCION 


LD BC, (0200) 
LD B, 08 

LD DE, (0202) 
LD D, 00 

LD HL, 0000 
SRLC 

JR NC,0114 
ADD HL, DE 
SLA E 

RLD 

DEC B 

JP NZ,010F 


8888888888 


Figura 3.24 
Un pase del bucle. 


La figura 3.40 recoge el listado completo de los contenidos 
de todos los registros y banderas del Z80. En la figura 3.41 
aparece un listado decimal y hexadecimal. 


OTRAS POSIBILIDADES DE PROGRAMACION 
El programa que acabamos de desarrollar puede escribirse 


de forma distinta. Como norma general, el programador debe 
estar en condiciones de modificar y mejorar cualquier progra- 
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ma. En este caso se ha desplazado el multiplicando antes de 
sumar, pero matemáticamente hubiera sido lo mismo desplazar 
el resultado una posición a la derecha antes de sumarlo al 
multiplicando. De hecho, se trata de un ejercicio interesante. 


Ejercicio 3.19: Escríbase un programa de multiplicación de 8 x 8 
con el mismo algoritmo ya visto, pero desplazando el resultado 
una posición a la derecha, en lugar de hacer lo mismo hacia la 
izquierda con el multiplicando. Compárese con el programa 
anterior, y determínese si el nuevo tratamiento será o no más 
rápido. Las velocidades de ejecución de las instrucciones del 
Z80 se recogen en el capitulo siguiente. 


PROGRAMA DE MULTIPLICACION MEJORADO 


El programa que acabamos de estudiar es una traducción 
directa del algoritmo. Sin embargo, para programar con eficacia 
hay que dedicar atención al detalle, para ver si puede acortarse el 
programa o acelerarse su velocidad de ejecución. Veamos ahora 
algunas opciones que mejoran el programa de multiplicación. 


Paso 1 


Una posibilidad de mejora consiste en aprovechar mejor las 
instrucciones del Z80. Así, las instrucciones penültima y antepe- 
nültima pueden reemplazarse por una sola: 

DJNZ MULT 


Se trata de una instrucción especial del Z80 de “salto 


. automático" que decrementa el registro B y bifurca a una 


posición determinada si no vale “0”. Hablando estrictamente, la 
instrucción no equivale por completo a las otras dos: 


DEC B 
JP NZ, MULT 


porque especifica un desplazamiento, y sólo puede darse un salto 
dentro del intervalo — 126 a + 129. Sin embargo, en este caso 
debemos saltar a una posición alejada tan sólo unos pocos 
bytes, por lo que la mejora es factible. El programa resultante 
aparece en la figura 3.25. 


MPY88B LD DE, (MPDAD) 


LD BC, (MPRAD) 
LD B, 8 CONTADOR DE 
BIT 
LD НГ, 0 
MULT SRL C 
JR NC, NOADD 
ADD HL, DE 
NOADD SLA E 
RL D 
Figura 3.25 DINZ MULT 
Programa de multiplicación LD (RESAD), HL 
mejorado, paso 1. RET 


Paso 2 


Como puede observarse en el programa inicial de la figura 
3.14, se emplean tres operaciones de desplazamiento diferentes: 
el multiplicador se desplaza a la derecha, a continuación se 
desplaza a la izquierda el multiplicando MPD en dos operacio- 
nes, un desplazamiento a la izquierda del registro E más una 
permutación circular a la izquierda del D. Todo esto consume 
tiempo. Hay un “truco” habitual en la programación de la 
multiplicación, que se basa en el hecho de que cada vez que el 
multiplicador se desplaza un bit a la derecha, en el registro 
multiplicador queda libre otro bit. Así, suponiendo que el 
desplazamiento se produce hacia la derecha —саѕо del ejemplo 
anterior— queda libre un bit a la izquierda. Se observa también 
que el primer producto parcial (o "resultado") utiliza, como 
máximo, 9 bits. Si al principio hubiésemos reservado para el 
resultado un solo registro, podríamos haber utilizado la posi- 
ción que deja libre el multiplicador para almacenar el noveno 
bit de aquél. 

Tras el siguiente desplazamiento de MPR, el tamaño del 
producto parcial vuelve a aumentar en un bit, de tal manera 
que puede reservarse al principio un solo registro para el 
producto parcial y utilizar la posición libre que se produce al 
desplazar МРК; por tanto, para mejorar el programa, asignare- 
mos MPR y RES a un par de registros. Lo ideal sería despla- 
zarlos juntos en опа sola operación, pero el 280 sólo es capaz 
de desplazar 8 bits de una vez; como la mayor parte de los 
microprocesadores de 8 bits, no dispone de las instrucciones 
necesarias para desplazar 16 bits. (ADD HL,HL desplazan los 
16 bits de HL una posición a la izquierda.) 

Pero queda otro recurso. El 780 —como el 8080— dispone 
de instrucciones especiales de adición de 16 bits que ya hemos 
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Figura 3.26 
Registros del programa mejora- 
do de multiplicación. 
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utilizado en este libro. Si el multiplicador y el resultado están 
almacenados en los registros apareados H y L, podemos usar la 
instrucción. 


ADD НЕ, HL 


que suma los contenidos de H y Га sí mismos. Sumar un 
número a si mismo equivale a duplicarlo, y en el sistema 
binario, duplicar un número equivale a desplazarlo hacia la 
izquierda una posición; por tanto, hemos realizado un desplaza- 
miento de 16 bits de una sola vez. Por desgracia, el desplaza- 
miento se ha producido hacia la izquierda y no hacia la 
derecha, como queríamos, pero no hay problema. 
Conceptualmente, MPR puede desplazarse tanto a la iz- 
quierda como a la derecha. Hemos utilizado el algoritmo de 
desplazamiento a la derecha, porque es el que se usa en la suma 
convencional, pero no hay necesidad de hacer así las cosas. La 
operación de suma es conmutativa y admite la inversión del 
orden, por lo que da lo mismo desplazar MPR a la izquierda. 
Para sacar partido a este desplazamiento simulado de 16 
bits tendremos que desplazar MPR a la izquierda, por lo que 
éste residirá en el registro H, y el resultado en el L. La 
configuración de registros resultante se ilustra en la figura 3.26. 


B| CONTADOR 


El resto del programa es básicamente igual al anterior. 
Aparece en la figura 3.27. 

Al comparar este programa con el anterior se observa que 
se ha reducido la longitud del bucle de multiplicación (el 
número de instrucciones entre MULT y el salto). Este progra- 


Figura 3.27 
Programa mejorado de multipli- 
cación, paso 2 


ma tiene menos instrucciones, y, en principio, avanzará con más 
rapidez, lo que demuestra la importancia de almacenar la 
información en los registros idóneos. 


MUL88C LD HL, (MPRAD-1) 
LD ‚0 
LD DE, (MPDAD) 
LD D, 0 
LD B, 8 CONTADOR 
MULT ADD НІ, НІ. DESPLAZA- 
MIENTO A LA 
IZQUIERDA 
JR NC, NOADD 
ADD  HL,DE 
NOADD DJNZ MULT 
LD (RESAD), HL 
RET 


El diseño de programas “directo” da, por lo general, resulta- 
dos que funcionan, pero que no son óptimos. Es, pues, impor- 
tante aprender a sacar el máximo partido a los registros e 
instrucciones disponibles. Los ejemplos que hemos visto consti- 
tuyen un enfoque racional de la selección de registros e instruc- 
ciones con vistas a optimizar la eficacia. 


Ejercicio 3.20: Calcúlese la velocidad de multiplicación con el 
último programa. Supóngase que tiene lugar una bifurcación 
en el 50 por 100 de los casos. El número de ciclos consumidos 
por cada una de las instrucciones aparece en el capítulo 
siguiente; la frecuencia del reloj será de 2 MHz (un ciclo 
= 0,5 ns). 


Ejercicio 3.21: Obsérvese que hemos utilizado el par de registros 
D y E para albergar el multiplicando. ¿Cómo sería el progra- 
ma anterior si hubiésemos utilizado el par B y C? ( Una pista: 
sería necesario hacer una modificación al final.) 


Ejercicio 3.22: ¿Por qué hay que molestarse en poner а 0 el 
registro D al cargar MPD en E? 


Por ültimo, vamos a prestar atención a un detalle que puede 
parecer irritante al programador no familizarizado con el Z80. 
Como habrá observado el lector, para cargar MPD en E a 
partir de la memoria es preciso cargar simultáneamente los dos 
registros D y E desde la dirección de memoria, porque, a menos 
que la dirección esté contenida en los registros H y L, no hay 
forma de traer un solo byte directamente y cargarlo en el 
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Figura 3.28 
Registros de la multiplicación 
de 16 x 16. 
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registro E; es una peculiaridad heredada del primitivo 8008, que 
carecía de direccionamiento directo. Con algunas mejoras, esa 
peculiaridad pasó al 8080, y mejoró todavía más en el Z80, en 
el que pueden traerse directamente 16 bits desde una dirección 
dada, pero no 8 bits, salvo hacia el registro A. 

Ahora, una vez solucionado este problema un tanto miste- 
rioso, pasaremos a programar una multiplicación más compli- 
cada. 


MULTIPLICACION DE 16 x 16 


Para poner a prueba todo lo que ya hemos aprendido, 
vamos a multiplicar dos números de 16 bits, aunque supondre- 
mos que el resultado precisa únicamente 16 bits para que quepa 
en un par de registros. 

Este, como en el primer ejemplo de multiplicación, pasará a 
los registros H y L (véase la figura 3.28). El multiplicando MPD 
reside en los registros D y E. 


CONTADOR MPR, SUP 


Es tentador situar el multiplicador en los registros B y C, 
pero si queremos aprovechar la instrucción DJNZ, el registro B 
debe reservarse para el contador; en consecuencia, la mitad del 
multiplicador estará en el registro C, y la otra mitad, en el A 
(véase la figura 3.28). El programa de multiplicación aparece en 
la figura 3.29. 


MUL16 ED A, (MPRAD + 1) MPR, SUPE- 


RIOR 
LD С, А 
LD A, (MPRAD) MPR, INFE- 
RIOR 
LD B, 16D CONTADOR 
LD DE, (MPDAD) MPD 
LD HL, 0 
MULT SRL С DESPLAZA- 
MIENTO 
DERECHA 
MPR, SUPE- 
RIOR 
RRA ROTACION 
CIRCULAR 
DERECHA 
MPR, INFE- 
RIOR 
JR NC, NOADD VERIFICAR 
ACARREO 
c74 ADD HL, DE SUMAR 
MPD AL RE- 
SULTADO 
NOADD EX DE, HL 
ADD HL, HL DOBLE 
DESPLAZA- 
MIENTO 
MPD IZ- 
QUIERDA 
Figura 3.29 r DE, HE 
Programa de multiplicación de DJNZ MULT 
16 x 16. RET 


El programa es análogo al que hemos desarrollado antes. 
Las primeras seis instrucciones (desde la etiqueta MULI6 a la 
MULT) inician los registros con los contenidos necesarios. El 
que las dos mitades de MPR deban cargarse en operaciones 
separadas constituye una complicación adicional. Se supone que 
MPRAD señala la parte inferior de MPR en la memoria; la 
parte superior ocupa la posición secuencial siguiente (natural- 
mente, puede utilizarse la convención contraria). Una vez leída 
la parte superior de MPR en A, debe transferirse a C: 


LD A,(MPRAD + 1) 
LD СА 
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Por último, la parte inferior de MPR puede leerse directa- 
mente en el acumulador: 


LD А, (МРВАР) 


El resto de los registros —B, О, Е, Ну L— se inician de la for- 
ma habitual: 


LD B, 16D 
LD DE,(MPDAD) 
LD HL, 


Hay que realizar un desplazamiento de 16 bits sobre el multi- 
plicador, lo que exige dos operaciones independientes de despla- 
zamiento o de rotación circular sobre los registros C y A: 


MULT SLR C 
RRA 


Tras el desplazamiento de 16 bits, el bit de la derecha de 
MPR, es decir, el BmS (bit menos significativo), ocupa el bit de 
acarreo C, en el que puede verificarse: 


JR NC, NOADD 


Como es habitual, el multiplicando no se suma al resultado si 
el bit de acarreo es “0”, pero sí se añade si es “1”. 


ADD HL,DE 


A continuación hay que desplazar el multiplicando MPD una 
posición hacia la izquierda. 

Sin embargo, el Z80 no dispone de ninguna instrucción que 
permita desplazar simultáneamente los contenidos de los regis- 
tros D y E una posición a la izquierda, y tampoco es posible 
sumar a sí mismos esos contenidos, que, por tanto, deben 
transferirse a H y L, duplicarse y devolverse otra vez a D y E. 
De todo esto se encargan las tres instrucciones siguientes: 


NOADD ЕХ DE, HL 
ADD HL,HL 
EX DE, HL 


Por ültimo, se reduce el contador B y se produce un salto al 
principio del bucle si esa reducción no lo lleva a “0”. 


DJNZ MULT 


Figura 3.30 
Multiplicación de 18 x 16 con 
resultado de 32 bits. 


División binaria 


Como siempre, pueden pensarse otras formas de organizar 
los registros para obtener, o no obtener, programas más cortos. 


Ejercicio 3.23: Cargar el multiplicador en los registros B y C, 
colocar el contador en A, escribir el programa de multiplica- 
ción correspondiente y discutir las ventajas o inconvenientes de 
esta organización de los registros. 


Ejercicio 3.24: En el programa original de multiplicación de 16 
bits de la figura 3.29, ¿habría alguna forma de desplazar 
MPD, contenido en los registros D y E, sin transferirlo a los 
H y L? 

Ejercicio 3.25: Escríbase un programa de multiplicación de 
16 x 16 bits que detecte resultados de más de 16 bits. Se trata 
de una sencilla mejora del programa básico. 


Ejercicio 3.26: Escríbase un programa de multiplicación de 
16 x 16 bits que admita resultados de 32 bits. La organiza- 
ción de registros sugerida aparece en la figura 3.30. Recuérde- 
se que el resultado inicial tras la primera suma del bucle sólo 
necesitará 16 bits y que el multiplicador dejará un bit libre por 
cada iteración posterior. 


+ P 
RESULTADO 
TRAS 

i й 


LA MULTIPLICACION 
Pasemos ahora a la última de las operaciones aritméticas 
usuales: la división. 


El algoritmo de la división binaria es análogo al utilizado 
para la multiplicación: el divisor se resta, sucesivamente, de los 
bits de orden superior del dividendo; tras cada resta, se usa el 
resultado en lugar del dividendo inicial; simultáneamente, se 
incrementa cada vez en 1 el valor del cociente. А veces, el 
resultado de la resta es negativo, y a esa situación se le llama 
sobrepasamiento; para solucionarla, se restaura el resultado par- 
cial, sumándole el divisor otra vez (naturalmente, hay que redu- 
cir simultáneamente en 1 el cociente). А continuación se despla- 
zan un bit a la izquierda el cociente y el dividendo, y se repite el 
algoritmo. El diagrama de flujo se ilustra en la figura 3.31. 
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Figura 3.31 
Diagrama de flujo de la división 
binaria de 8 bits. 


Figura 3.32 
Registros de la división 16/8. 
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INICIAR 
COCIENTE = 0 
CONTADOR DESPL =4 


DESPLAZAR 120 
DIVIDENDO (CON 8 
CEROS INICIALES) 

Y COCIENTE 


RESTA ОЕ PRUEBA IZQUIERDA 
(DIVIDENDO) - DIVISOR 


RESTAURAR: 
SUMAR DIVISOR 


FIN (RESTO A LA IZQUIERDA (DIVIDENDO)) 


Este método es el llamado de restauración. Hay una variante 
de ejecución más rápida llamada sin restauración. 
DIVISION 16 POR 8 


Examinemos, como ejemplo, una división de 16 x 8, que 
deja un cociente de 8 bits y un resto del mismo formato. La 
figura 3.32 recoge la disposición de los registros. 


B [CONTADOR c 


H 


Figura 3.33 
Programa de división 16/8. 


El programa aparece en la figura 3.33: 


DIV168 LD 
LD 
LD 
LD 
LD 


DIV XOR 
SBC 


INC 


JP 


ADD 
DEC 


NOADD ADD 


DJNZ 
RET 


А, (DVSAD) 
D, A 

E,0 

HL, (DVDAD) 
B,8 


A 
HL, DE 


HL 


P, NOADD 


HL, DE 
HL 


HL, HL 


DIV 


CARGAR DIVI- 
SOR 
EN D 


CARGAR DIVI- 
DENDO DE 16 
BITS 
INICIALIZAR 
EL CONTADOR 
BORRAR BIT C 
DIVIDEN- 

DO - DIVISOR 
COCIENTE = 
COCIENTE + 1 
VERIFICAR SI 
EL RESTO ES 
POSITIVO 
RESTAURAR SI 
ES NECESARIO 
COCIENTE = 
COCIENTE - 1 
DESPLAZAR 
DIVIDENDO A 
LA IZQUIERDA 
BUCLE HASTA 
QUE B- 0 


Las primeras cinco instrucciones cargan el divisor y el divi- 
dendo, respectivamente, en los registros correspondientes, y, 
además, inicializan el contador, en el registro B, al valor 8. 
Obsérvese que el registro B constituye el alojamiento idóneo del 
contador cuando se utiliza la instrucción especial del 280, 


DJNZ: 
DIV168 LD A,(DVSAD) 
LD D,A 
LD E 
LD H 
LD B 


A continuación se resta el divisor del dividendo. Como hay 
que usar una instrucción SBC (no hay resta de 16 bits sin 
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acarreo), es preciso poner el acarreo а “О” antes de realizar la 
operación, cosa que puede hacerse de varias formas; el acarreo 
puede anularse con instrucciones como las siguientes: 

XOR A 

AND A 

OR A 


En este caso se ha utilizado XOR: 
DIV XOR A 

Ahora puede efectuarse la resta: 
SBC HL, DE 


Se anticipa que la resta dejará un resto positivo, paso que se 
conoce como “resta de prueba” (véase el diagrama de flujo de la 
figura 3.31); por tanto, el cociente se incrementa en-1. Si la resta 
falla (es decir, si el resto es negativo), será preciso reducir a 
continuación en 1 el cociente: 


INC HL 
Se verifica el resultado de la resta: 
JP P,NOADD 


Si el resto es positivo o 0, es que el resultado ha sido 
correcto, y no es preciso almacenarlo. El programa salta a la 
dirección NOADD. En caso contrario, el dividendo en curso 
debe ponerse con su valor anterior sumándole el divisor, a la 
vez que se resta 1 al cociente. Las siguientes instrucciones se 
encargan de ejectuar estos pasos: 


ADD HL,DE 
DEC HL 


Por ültimo, se desplaza a la izquierda el dividendo resultan- 
te, como anticipación a la siguiente resta de prueba. Se reduce 
el contador B y se comprueba si vale "0". Mientras esto no 
ocurra, se ejecuta el bucle: 


NOADD ADD HL,HL 
DJNZ DIV 
RET 


Ejercicio 3.27: Verifíquese manualmente el funcionamiento de este 
programa de división cumplimentando la tabla de la figura 
3.34, tal como se hizo en el ejercicio 3.18 con la multiplicación. 
Obsérvese que no es preciso introducir en esta tabla el conte- 
nido de D, porque no se modifica nunca. 


Figura 3.34 
Tabla para el programa de divi- 
sión. 


DIVISION DE 8 BITS 


El programa que se propone aquí sigue un procedimiento de 
restauración, y deja en Á un cociente complementado. Sirve 
para efectuar divisiones de 8 bits por 8 bits sin signo. 


E ES EL DIVIDENDO 
C ES EL DIVISOR 

A ES EL COCIENTE 
B ES EL RESTO 


DIV88 XOR A BORRAR EL ACU- 
MULADOR 
LD В, 8 CONTADOR DEL 
BUCLE 


CIRCULAR DE CY 
EN ACC-DIVIDEN- 


LOOP88 RL E PERMUTACION 


DO . 
RLA SALIDA DE CY 
SUB C DIVISOR DE LA 
RESTA DE PRUE- 
BA 
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Figura 3.35 
Registros de la división sin res- 
tauración. 
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JR NC, $ + 3 RESTA CORRECTA 
ADD А, С RESTAURAR 
ACUM, PONER CY 


DJNZ LOOP88 


LD B,A PONER EL RESTO 
EN B 

LD А, Е OBTENER EL CO- 
CIENTE 

RLA DESPLAZAR EL 

. ULTIMO BIT DEL 

RESULTADO 

CPL BITS DE COM- 
PLEMENTO 

RET 


Nota: El símbolo *$" de la sexta instrucción representa el 
valor del contador de programa. 


DIVISION SIN RESTAURACION 


El programa siguiente lleva a cabo una división de un 
entero de 16 bits por otro de 15 bits mediante una técnica sin 
restauración. IX sefiala el dividendo e IY el divisor (no cero). 
(Véase la figura 3.35.) 


А | DVD, SUP 
B | CONTADOR DVD, INF | С 


IX DIRECCION DVD 
IY DIRECCION DVS 


El registro B, inicialmente de valor 16, es el contador. 

A y C contienen el dividendo. 

D y E contienen el divisor. 

H y L contienen el resultado. 

El dividendo de 16 bits se desplaza hacia la izquierda me- 
diante las instrucciones: 


RL С 
RLA 


El resto se desplaza hacia la izquierda mediante la intruc- 
ción: 


ADC HL, HL. 


El cociente final queda en B, C y el resto en HL. El progra- 
ma continúa. 


DIV16 LD B, (IX + 1) 
LD C, (IX) 
LD D, (IY + 1) 
LD E, (ТҮ) 
LD A, D 
OR E PARTE SUPE- 
RIOR DEL (DI- 
VISOR) O PAR- 
TE INFERIOR 
| DEL (DIVISOR) 
JR Z, ERROR VERIFICAR SI 
| DIVISOR =0 
LD A, B OBTIENE 
(DVD) SUP 
LD HL,0 BORRAR RE- 
SULTADO 
LD B, 16D CONTADOR 
TRIALSB RL C ROTACION 
CIRCULAR RE- 
SULTADO 
+ ACC IZQ 
RLA 
ADC НІНІ DESPLAZAR A 
LA  IZQUIER- 
DA. NO PONE 
ACARREO 
SBC HL, DE MENOS DIVI- 
SOR 
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NULL ССЕ BIT DE КЕ- 


SULTADO 
JR NC, NGV ¿ACUMULA- 
DOR NEGATI- 
VO? 
PTV DJNZ TRIALSB ¿CONTADOR 
CERO? 
ТР DONE 
RESTOR RL C ROTACION 
CIRCULAR RE- 
SULTADO 
+ ACC IZQ 
RLA 
ADC НІНІ. COMO ARRI- 
BA 
AND A 
ADC  HL,DE RESTAURAR 
SUMANDO 
DVSR 
JR C, PTV RESULTADO 
POSITIVO 
JR Z, NULL RESULTADO 
CERO 
NGV DJNZ RESTOR ¿CONTADOR 
CERO? 
DONE RL C DESPLAZAR 
BIT DE RE- 
SULTADO 
RLA 
ADD  HL,DE RESTO CO- 
RRECTO 
LD B, A COCIENTE EN 
B,C 
RET 


Ejercicio 3.28: Compárese el programa anterior con el siguiente, 
que utiliza una técnica de restauración: 


DIVIDENDO EN AC 
DIVISOR EN DE 
COCIENTE EN AC 
RESTO EN HL 


DIV16 LD HL,O BORRAR ACU- 
MULADOR 
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LD 


LOOP16 RL 


DJNZ 


RL 


RLA 
RET 


B, 16D 


NC, $ +3 


HL, DE 


LOOP16 
C 


PONER CON- 
TADOR 
ROTACION 
CIRCULAR 
ACC-RESUL- 
TADO 


DESPLAZA- 
MIENTO A LA 
IZQUIERDA 
DIVISOR RES- 
TA DE PRUE- 
BA 

RESTA CO- 
RRECTA 
RESTAURAR 
ACUMULA- 
DOR 
CALCULAR 
BIT DEL RE- 
SULTADO 

EL CONTADOR 
NO ES CERO 
DESPLAZAR 
EL ULTIMO 
BIT DEL RE- 
SULTADO 


Nota: El símbolo “$” de la séptima instrucción significa 


“posición en curso”. 


Operaciones lógicas 


La otra clase de instrucciones que puede ejecutar la ALU 
son las instrucciones lógicas: AND, OR y OR exclusivo (XOR). 
También podrían incluirse aquí las ‘operaciones de desplaza- 
miento y rotación circular, utilizadas ya repetidamente, y la 
instrucción de comparación, que en el Z80 se llama CP. El uso 
individual de AND, OR y XOR se describirá en el capítulo 4. 

Vamos ahora a desarrollar un breve programa para com- 
probar si una posición de memoria dada llamada LOC contiene 
el valor “0”, el “1” o algún otro. 
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Utilizaremos en el programa la instrucción de comparación 
y realizaremos una serie de comprobaciones lógicas. Según el - 
resultado de la comparación, se ejecutará uno u otro segmento 
del programa. 

El programa es éste: 


LD A,(LOC) LEER EL CARAC- 


TER DE LOC 
СР 00H COMPARAR CON 
0 
JP Z, CERO ¿ES À 0? 
CP ОН COMPARAR CON 
І 
ЈР Z UNO 
NOENCONTRADO ... 
CERO m 
UNO 


La primera instrucción, “LD А, (ГОС)”, lee el contenido de 
la posición de memoria LOC y la carga en el acumulador. El 
contenido es el carácter que deseamos comprobar, y su valor se 
compara con 0 mediante la instrucción: 


CP 00H 


La instrucción compara el contenido del acumulador con el 
valor hexadecimal “00”, es decir, con el binario “0000 0000”. 
Esta instrucción de comparación pone el bit Z del registro de 
estado al valor “1”, si el resultado es afirmativo, lo que se 
comprueba mediante la instrucción: 


JP Z CERO 


La instrucción de salto comprueba el valor del bit Z. Si el 
resultado de la comparación ha sido positivo, Z valdrá uno y se 
efectuará el salto a la dirección CERO. Si el resultado es negati- 
vo, se ejecutará la instrucción siguiente de la secuencia: 


CP ОН 


De la misma manera, la instrucción de salto bifurcará a la 
posición UNO si la comparación es positiva. Si ninguna de las 
comparaciones fuesen positivas, se ejecutaría la instrucción que 
ocupa la posición NOENCONTRADO. 


JP 2, UNO 
NOENCONTRADO... 


La finalidad de este programa es poner de relieve el valor 
de la instrucción de comparación seguida de salto. Esta combi- 
nación podrá utilizarse en muchos de los programas que vienen 
a continuación. 


Ejercicio 3.29: Búsquese en el capítulo siguiente la definición de la 
instrucción LD A,(LOC) y examinese su efecto sobre las 
banderas en caso de que hubiere alguno. ¿Es imprescindible la 
segunda instrucción de este programa (CP 00H )? 


Ejercicio 3.30: Escríbase un programa que lea el contenido de la 
posición de memoria “24” y bifurque a la dirección llamada 
“ESTRELLA”, si en dicha posición se encuentra el simbolo 
“ж”, La representación binaria de “+” es "00101010". 


Resumen de instrucciones 


Subrutinas 


Hemos estudiado casi todas las instrucciones importantes 
del Z80 utilizándolas; hemos transferido valores entre la memo- 
ria y los registros; hemos realizado operaciones aritméticas y 
lógicas con esos valores; los hemos verificado, y, segün el resul- 
tado obtenido, hemos ejecutado unas u otras porciones del 
programa. También hemos aprovechado las instrucciones “au- 
tomáticas" especiales del Z80, como DJNZ, para acortar pro- 
gramas. Más adelante recurriremos a otras instrucciones auto- 
máticas, como LDDR, CPIR o INIR. 

Se ha sacado el máximo partido de las características pecu- 
liares del Z80, como las instrucciones para registros de 16 bits 
que simplifican los programas (téngase en cuenta que tales 
características no existen en el 8080, del que el Z80 es una 
versión optimizada). 

Ya hemos hablado de una estructura llamada bucle, y a 
continuación estudiaremos otra muy importante: la subrutina. 


Conceptualmente, una subrutina no es De un bloque de 
instrucciones al que el programador ha adjudicado un nombre. 
Desde un punto de vista práctico, toda subrutina empieza con 
una instrucción especial, llamada declaración de subrutina, que 
la identifica como tal al ensamblador, y' termina con otra, 
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Figura 3.36 


Llamadas a una subrutina. 
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llamada retorno (return). Veremos primero el funcionamiento 
de una subrutina dentro de un programa, para aprender a 
apreciar su valor, y a continuación pasaremos a la realización 
práctica de la misma. 

La utilización de una subrutina se muestra en la figura 3.36. 
El programa principal está representado a la izquierda, y la 
subrutina, a la derecha. Las líneas del primero se ejecutan una 
tras otra hasta que aparece una instrucción “CALL SUB” o 
llamada a subrutina, que transfiere el control a ésta, de manera 
que la primera instrucción que se ejecuta tras CALL SUB es la 
primera de las que componen la subrutina en cuestión, como 
muestra la flecha 1 de la figura. 


PROGRAMA PRINCIPAL 


SUBRUTINA 


RETORNO 


А continuación se ejecuta el subprograma de la subrutina de 
la. misma forma que cualquier otro programa. Supondremos еп 
principio que la subrutina no incluye, a su vez, otras llamadas. 
La última instrucción de la subrutina es RETORNO, lo que 
provoca la devolución del control al programa principal. Tras 
dicha instrucción, la primera que se ejecuta es la que sigue a 
CALL SUB en el mencionado programa principal, hecho que 
muestra la flecha 3 de la figura. А continuación prosigue de la 
forma normal la ejecución del programa principal (flecha 4). 

Dentro del programa principal surge una segunda instruc- 
ción CALL SUB, y tiene lugar una segunda transferencia, sim- 
bolizada por la flecha 5. Esto significa que la subrutina vuelve a 
ejecutarse, una vez más, tras la nueva llamada. 

En el momento en que aparece la instrucción RET 
dentro de la subrutina se ejecuta la instrucción del programa 
principal que sigue a la CALL SUB, como muestra la flecha 7, 
y tras ella el resto de dicho programa (flecha 8). 

El efecto de las instrucciones especiales CALL SUB y КЕТ. 
está, por tanto, claro. Falta por saber qué utilidad tienen las 
subrutinas. 

Lo más valioso de una subrutina es que puede llamarse 
desde cualquier punto del programa principal y utilizarse cuan- 
tas veces sea necesario sin necesidad de volver a escribirla. De 


esta forma se ahorra espacio de memoria y tiempo de progra- 
mación, con la consiguiente simplificación del diseño de pro- 
gramas. 


Ejercicio 3.31: ¿Cuál es el principal inconveniente de la subrutina? 


Respuesta: El inconveniente del trabajo con subrutinas se deduce 
fácilmente con sólo examinar el flujo de control entre ellas y 
el programa principal: la velocidad general de ejecución es más 
baja, porque es preciso ejecutar las instrucciones especiales 
CALL SUB y RETURN. 


REALIZACION PRACTICA DEL MECANISMO DE LA 
SUBRUTINA 


Vamos a ver de qué forma se tratan en el interior del 
microprocesador las dos instrucciones especiales CALL SUB y 
ВЕТ. El efecto de CALL SUB es tomar la instrucción si- 
guiente de una nueva dirección. Como se recordará (y si no se 
recuerda deberá repasarse el capítulo 1), la dirección de la 
instrucción que debe ejecutarse a continuación de la que está en 
curso se encuentra en el contador del programa PC. De esto se 
deduce que CALL SUB modifica el contenido del PC, en el que 
carga la dirección de comienzo de la subrutina. Pero, ¿basta con 
eso? 

Para responder a esa pregunta, consideremos la segunda 
de las instrucciones especiales: RET. Esta instrucción deter- 
mina la vuelta a la instrucción que sigue a CALL SUB, lo que 
sólo es posible si su dirección se ha conservado en algún sitio. 
Dicha dirección es el valor del contador del programa en el 
momento en que se llega a CALL SUB, porque el contador del 
programa se incrementa automáticamente cada vez que se usa 
(repasar el capítulo 1). Esta es precisamente la instrucción que 
debe conservarse para ejecutar más adelante RET. 

El problema que se plantea es dónde conservar esa dirección 
de retorno, que debe permanecer en un sitio en el que no pueda 
borrarse de ninguna manera. 

Antes de seguir, vamos a analizar la situación que plantea la 
figura 3.37: la subrutina 1 contiene una llamada a otra subruti- 
na 2 o SUB2. El mecanismo expuesto debe funcionar también 
en este caso. Naturalmente, el número de llamadas internas no 
tiene por qué estar limitado a dos, y puede ser cualquiera N. En 
general, cada vez que se encuentre una nueva llamada CALL, el 
mecanismo de ejecución debe volver a almacenar el contador 
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Figura 3.37 
Llamadas internas. 
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del programa, lo que significa que hacen falta al menos 2N 
posiciones de memoria para este mecanismo. Además, hay que 
volver, primero, desde SUB2 y, a continuación, desde SUBI. En 
otras palabras, lo que hace falta es una estructura capaz de 
conservar el orden cronológico en que se han almacenado las 
direcciones. 

Dicha estructura tiene un nombre, y ya hemos hablado de 
ella: es la pila. La figura 3.39 recoge el contenido real de la pila 
durante las sucesivas llamadas a las subrutinas. Examinemos 
primero el programa principal. La primera llamada se encuen- 
tra en la dirección 100: CALL SUBI. Supongamos que, en el 
microprocesador, la llamada a la subrutina utiliza 3 bytes (RST 
es una excepción); por tanto, la siguiente dirección secuencial 
no es “101”, sino “103”; la instrucción CALL utiliza las direc- 
ciones “100”, “101” y “102”; como la unidad de control del Z80 
“sabe” que se trata de una instrucción de 3 bytes, el valor del 
contador del programa cuando la llamada haya sido decodifica- 
da en su totalidad será “103”. El efecto de la llamada será 
cargar el valor “280” —dirección de partida de SUBI— en el 
contador del programa. 


, 


PROGRAMA PRINCIPAL 


SUB 2 


` RETORNO 


Ya estamos en condiciones de estudiar el efecto de la ins- 
trucción RET y el funcionamiento del mecanismo de la 
pila. La ejecución avanza dentro de SUB2 hasta que encuen- 
tra la instrucción RET en el momento 3. El efecto de RET 
no es sino transferir la cabecera de la pila al contador del 
programa. En otras palabras, el contador recupera el valor 
que tenía antes de la entrada a la subrutina. La parte superior 
de la pila es, en nuestro ejemplo, “303”. La figura 3.39 indica 
que, en el momento 3, el valor “303” ha pasado de la pila al 
contador del programa. Como resultado, la ejecución de la 
instrucción avanza a partir de la dirección “303”. En el ciclo 4 
se enċuentra la instrucción RET de SUB1. El valor de la 
cabecera de la pila es “103”, que pasa al contador del progra- 
ma. Como consecuencia, el programa se ejecuta, a partir de la 
posición de memoria “103”, dentro del programa principal, que 
es precisamente el efecto deseado. La figura 3.39 demuestra que 
en el ciclo 4 la pila está de nuevo vacia. El mecanismo funciona. 


CALL SUB 


SUB 
CALL SUB 2 
RETORNO + 


Figura 3.38 


Llamadas a subrutinas. 


Figura 3.39 
Estado de la pila a lo largo del 
tiempo. 


Este mecanismo de llamada a subrutinas actúa hasta que la ` 
pila alcanza su dimensión máxima, у por eso los primitivos mi- ` 
croprocesadores con pilas de 4 u 8 registros estaban limitados a 
4 u 8 niveles de llamadas a subrutinas. 

Obsérvese que en las figuras 3.37 y 3.38 las subrutinas se 
han simbolizado a la derecha del programa principal; ello obe- 
dece únicamente a razones de claridad de la representación, 
porque las subrutinas se escriben exactamente igual que las 
instrucciones normales del programa. En la hoja de papel en 
que aparece el listado completo del programa, las subrutinas 
pueden colocarse al principio del texto, en el centro del mismo 
o al final, y se identifican por la declaración de subrutina que 
las precede. Las instrucciones especiales indican al ensamblador 
que lo que sigue debe tratarse como una subrutina. Estas seu- 
doinstrucciones del ensamblador se estudiarán en el capítulo 10. 


DIRECCION (PROGRAMA PRINCIPAL) 


o 
> 
= 
r 
o 
c 
о 


(SUB 1) 


CALL SUB 2 


RETORNO 


RETORNO 


ШШШ 


PILA: 


SUBRUTINAS DEL 780 


Ya se han expuesto los conceptos básicos relativos a las 
subrutinas. Sabemos que hace falta una pila pará que funcio- 
nen. El Z80 dispone de un puntero de pila de 16 bits, de 
manera que la pila puede residir en cualquier lugar de la me- 
moria y albergar hasta 64K (1K = 1024), suponiendo que 
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estén disponibles para ese fin. En la práctica, el programador 
define, antes de escribir el programa, la dirección de partida de 
la pila y su dimensión máxima, reservándose, en consecuencia, 
la parte necesaria de la memoria. 

La instrucción de llamada a subrutinas del Z80 es CALL, y 
existe en dos versiones: llamada directa o incondicional 
—CALL DIRECCION—, que es la que ya se ha descrito, y 
llamada condicional, peculiar del Z80, y en virtud de la cual se 
llama una subrutina si se satisface determinada condición. Por 
ejemplo, CALL NZ, SUBI llamará la subrutina 1 si la bandera 
Z es 0 en el momento de la verificación. Es una instrucción 
potente, porque muchas llamadas a subrutinas son condiciona- 
les, es decir, sólo se producen si se cumple una condición 
específica. 

CALL CC,NN sólo se ejecuta si es cierta la condición 
especificada por “CC”; CC es un conjunto de tres bits (bits 3, 4 
y 5 del código de operación) capaz de especificar hasta ocho 
condiciones, que corresponden a cada una de las cuatro bande- 
ras “Z”, "C", *P/V" y “S”, que pueden ser cero o no cero. 

Hay también dos tipos de instrucciones de vuelta: RET y 
RET CC. 

RET es la instrucción de vuelta básica. Ocupa un byte, y 
hace que los dos bytes superiores de la pila vuelvan a instalarse 
en el contador del programa. Es incondicional. 

RET CC tiene el mismo efecto, pero sólo se ejecuta si las 
condiciones especificadas por CC son ciertas. Los bits condicio- 
nales son los mismos de la instrucción CALL que acaban de 
describirse. 

Además, hay dos tipos especializados de retorno que sirven 
para acabar rutinas de interrupción: RETI y RETN. Se descri- 
birán en el capítulo de instrucciones del Z80 y en el de interrup- 
ciones. 

Hay, por fin, otra instrucción especializada análoga a una 
llamada a subrutina, pero que sólo permite al programa des- 
viarse a una de ocho posiciones de partida localizadas en la 
página cero. Se trata de la instrucción RST P, una instrucción 
de 1 byte que almacena automáticamente el contador del pro- 
grama en la pila y desvía el programa a la dirección especifica- 
da en el campo de tres bits P; éste corresponde a los bits 3, 4 y 
5 de la instrucción, multiplicados por ocho. 

En otras palabras, si los bits 3, 4 y 5 son "000", el salto se 
producirá a la posición 00H. Si son “001”, el salto será a 08H, 
etcétera, y así hasta 111, que provoca la bifurcación a la posición 
38H. La instrucción RST es muy eficaz en términos de veloci- 
dad, porque tiene un solo byte, aunque a cambio de saltar 
ünicamente a ocho posiciones en la página cero; además, estas 


direcciones de la página cero están separadas nada más que por 
ocho bytes. Se trata de una instrucción procedente del 8080 que 
se usa mucho para interrupciones, como se describirá en el 
capítulo correspondiente. No obstante, el programador puede 
utilizarla para cualquier otro fin, y debe considerarse como una 
posible llamada a una subrutina especializada. 


EJEMPLOS DE SUBRUTINAS 


Casi todos los programas desarrollados hasta el momento, y 
la mayor parte de los que vamos a desarrollar, se escribirían 
normalmente como subrutinas. Así, el programa de multiplica- 
ción es normal que se use en muchos puntos de un programa 
general; por tanto, para clarificar y facilitar el desarrollo de 
programas, conviene definir una subrutina llamada, por ejem- 
plo, MULT; al final de la misma no hay más que añadir la 
instrucción RET. 


Ejercicio 3.32: Si se usa MULT como subrutina, ¿“dañará” algu- 
nos de los registros o banderas internos? 


RECURRENCIA 


Se llama recurrencia a la llamada a una subrutina desde ella 
misma. Si se ha comprendido el mecanismo de ejecución prácti- 
ca de subrutinas, podrá responderse a la siguiente pregunta: 


Ejercicio 3.33: ¿Es posible que una subrutina se llame a sí misma? 
(En otras palabras, ¿funcionará todo correctamente si una sub- 
rutina se llama a sí misma?) Si no está seguro de la respues- 
ta, dibuje la pila y ocüpela con las direcciones sucesivas; 
observe a continuación los registros y la memoria ( véase ejer- 
cicio 3.18) y determine si hay algún problema. 


Las interrupciones se estudiarán en el capítulo 6, dedicado a 
las técnicas de entrada y salida. Todos los retornos, con excep- 
ción de los que proceden de interrupciones, son instrucciones de 
un byte; por su parte, todas las llamadas —excepto RST— son 
instrucciones de tres bytes. 


Ejercicio 3.34: Consulte en el capítulo siguiente los tiempos de 
ejecución de las instrucciones CALL y RET. ¿Por qué el 
retorno de una subrutina es mucho más rápido que la llamada 
a la misma? (Una pista: si la respuesta no parece obvia, 
repásese una vez más el funcionamiento de la pila del mecanis- 
mo de subrutinas y analícense las operaciones internas que 
deben llevarse a cabo.) 
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PARAMETROS DE SUBRUTINAS 


Cuando se llama a una subrutina, normalmente se espera 
que actúe sobre ciertos datos. Así, en el caso de la multiplica- 
ción, hay que transmitir a la subrutina dos números para que 
sean sometidos a esa operación. Como ya vimos en el caso de 
la rutina de multiplicación, el multiplicando y el multiplicador 
se encuentran en posiciones de memoria dadas. He aquí, pues, 
un procedimiento de paso de parámetros: a través de la memo- 
ria. Pero hay, además, otras dos técnicas, lo que da lugar a tres 
métodos: 


1. A través de los registros. 
2. A través de la memoria. 
3. A través de la pila. 


Usar los registros para transferir parámetros tiene la ventaja, 
suponiendo que haya registros disponibles, de que no es preciso 
trabajar con posiciones fijas de memoria, de manera que la 
subrutina es independiente de la memoria. Si se utiliza una 
posición de memoria fija, cualquier otro usuario del programa 
deberá tener mucho cuidado para seguir la misma convención y 
asegurarse de que esa posición está realmente libre (véase el. 
ejercicio 3.19). Por eso, en muchos casos, se reserva un bloque 
de posiciones de memoria para transferir parámetros entre va- 
rias subrutinas. 

Usar la memoria tiene la ventaja de la flexibilidad (pueden 
usarse más datos), pero a cambio de un rendimiento inferior y 
de tener que ligar la subrutina a un área fija de la memoria. 

La ubicación de los parámetros en la pila tiene la misma 
ventaja que el uso de los registros: la independencia de la 
memoria. La subrutina “sabe” que recibirá, por ejemplo, dos 
parámetros almacenados en la cabecera de la pila. Por supues- 
to, también tiene inconvenientes: la pila se satura de datos, con 
la consiguiente reducción del número de niveles de llamadas a 
subrutinas. También complica considerablemente el manejo de 
la pila, y puede exigir el empleo de varias de estas estructuras 
de datos. 

La elección es responsabilidad del programador; pero, en 
general, se prefiere conservar la mayor independencia posible 
con respecto a las posiciones reales de memoria. 

Si no hay registros disponibles, la pila es una alternativa a 
considerar. Sin embargo, cuando es necesario pasar a la subru- 
tina mucha información, ésta puede residir directamente en 
memoria. Una forma elegante de resolver el problema de trans- 
mitir un bloque de datos consiste simplemente en transmitir un 
puntero dirigido a la información (un puntero es la direc- 


Resumen 


ción del principio del bloque). El puntero puede pasarse por 
medio de un registro, o de la pila (hacen falta dos posiciones de 
pila para almacenar una dirección de 16 bits), о bien por medio 
de una o varias posiciones fijas de memoria. 

Por último, si ninguna de las dos soluciones es aplicable, 
habrá que acordar con la subrutina una posición fija en memo- 
ria (el “buzón de correos”). 


Ejercicio 3.35: ¿Cuál de los tres métodos mencionados será mejor 
para las recurrencias? 


BIBLIOTECA DE SUBRUTINAS 


La organización de las diversas porciones de un programa 
en forma de subrutinas identificables tiene la ventaja de que 
pueden ponerse a punto independientemente y de que se les 
puede asignar un nombre mnemotécnico. Si pueden utilizarse en 
segmentos diferentes del programa serán intercambiables y, por 


_ tanto, podrán formar parte de una biblioteca de subrutinas. Sin 


embargo, en programación no existe ninguna panacea, y acos- 
tumbrarse a convertir cualquier grupo de instrucciones que se 
repita por su función en una subrutina dará lugar a un rendi- 
miento escaso. El programador deberá aprender a equilibrar las 
ventajas con los inconvenientes. 


Hemos visto en este capítulo cómo manipulan internamente 
la información las instrucciones del Z80. Los algoritmos tradu- 
cidos a programas se han hecho cada vez más complicados y, 
además, han servido para utilizar y explicar los tipos de instruc- 
ciones más importantes. 

También hemos definido varias estructuras de uso continuo, 
como bucles, pilas y subrutinas. 

El lector deberá tener ya una idea básica de lo que es 
programar y de las principales técnicas puestas en juego en las 
aplicaciones normales. Pasemos, pues, a estudiar en detalle cada 
una de las instrucciones. 
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HL.=0000 
Н’=0000 
HL.=0000 
Н”ж0000 
HL=0000 
H”=0000 
HL.=0000 
H'x0000 
3 HL«0000 
Н”х0000 
НІ: <0000 
1559009 


HL: 0005 © 
H'x0000 


х=0000 
` 820300 


Үз0000 
"0116 


Figura 3.40 
Listado completo de la multipli- 
cación. 


Hs 0000 X«0000 Ys -0000 
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01197 
1:00 


` OLOF’ 


1=00 
0111“ 
1:00 
0113” 


1:00 
0118” 
1:00 
0119” 
1:00 
O10F* 
Те: 00, 


1:00 
0116” 
1=00 
01 18^ 
1:00 
0119” 
1=00 


LD 
en 
Ln 


Ln 


nEC 


JF 


ВС, (0200) 
(0200*) 
Er08 


DE + (0202) 
(020299 
П»,00 


НІ. » 0000 
(000079 
с 


NC»0114 
(01149 
HL.» DE 

Е 

n 

B 
NZ»010F 
(O10F^) 
E 

№, 0114 


(011499 
HL s DE 


NC:0114 
4011499 
E 


n 


NC»0114 
(01147) 
E 
n 
E 


NZyOLOF 
(OT10F*) 


Figura 3.40 
Listado completo de la multipli- 
cación (continuación). 


Figura 3.41 
Programa de 
(Hex). 


multiplicación 


N HL.=000F 7 O10F^ SRL. E 
т 1:00 
29 01117 JR NC»0114 
(011499 
Е 
n 
R 
Z010F 
€010F ^) 
C 
NC»0114 
(01147) 
DE=0140 E 
D':s0000 
0116 А п 
“0000 1:00 
01187 ПЕС В 
1500 
01197 JF Z»010F 
(OLOF?) 
C 
МС»О114 
(011499 
Е 
{=00 
01167 KL n 
1=00 
0118” ПЕС E 
1:00 
QYL9- JF М2»010Ғ 
1200 (O10F^) 
DE=0500 2 011€” LD (0204) » HL. 
1'z0000 1:00 (020479 
| 000F " O11F' МОР 
E”:=0000 1720000 Н’=0000 Х=0000 Y::0000 1:00 


RESPUESTAS AL EJERCICIO 3.18 (MULTIPLICACION): 


0100 10 


ORG #0100 

0100 0002 20 MPRAD DEFW #0200 
0102 0202 50 MPDAD DEFW #0202 
0104 0402 40 RESAD DEFW #0204 

50 3 
0106 Е0480001 60 MP488 LD BC, (MPRAD) ¿CARGA MULTIPLICADOR EN C 
010A 0608 70 LD в, в 3B ES CONTADOR DE BIT 
O10C EDSBO201 80 LD DE, (MPDAD) $CARGA MULTIPLICANDO EN E 
0110 1600 90 LD D,0 3INICIALIZA D 
0112 210000 100 LD HL,0 ¿PONE A O EL RESULTADO 
0115 CB39 110 MULT SRL C ¿SHIFT AL ACARREO DEL BIT 

1155 3 MULTIPLICADOR 
0117 3001 120 JR NC, NOADD $ COMPRUEBA EL ACAREO 
0119 19 130 ADD HL,DE 3SUMA AL RESULTADO MPD 
011A CB23 140 NOADD SLA E ¿SHIFT IZQUIERDA DE MPD 
O11C CB12 150 RL D ¿GUARDA EL BIT EN D 
011E ОЗ 160 DEC B 3DECREMENTA EL CONTADOR DE SHIFT 
ОЩЕ С21501 170 JP NZ,MULT ¿REPETIR SI CONTADOR<>0 
0122 220401 180 LD (RESAD) , HL j ALMACENA EL RESULTADO 
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ЕЛІН 


ETIQUETA [INSTRUCCION GETI 
(ACARREO) 
00 00 0 


00 00 | 00 
MP488 LD BC, (0200) 03 0 00 00 | 00 
LD B, 08 03 0 00 00 | 00 

LD DE, (0202) 03 0 00 00 | 00 

LD D, 00 03 0 00 00 | 00 

LD HL,0000 03 0 00 00 | 00 

SRLC 01 ] 00 00 | 00 

JR NC,0114 01 ] 00 00 | 00 

ADD HL, DE 01 0 00 00 | 05 

SLAE 01 0 00 00 | 05 

RLD 01 0 00 00 | 05 

DEC B 01 0 00 00 | 05 

JP NZ,O10F 01 0 00 00 | 05 

SRLC 00 1 00 00 | 05 

JR NC,0114 00 1 00 00 | 05 

ADD HL,DE 00 0 00 00 | OF 

SLA E 00 0 00 00 | OF 

RL D 00 0 00 00 | OF 

DEC В 00 0 00 00 | OF 

Figura 3.42 JP NZ,OTOF 00 0 00 00 | OF 


Dos repeticiones del bucle. 
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bd 
E odd 
Ld 
Ld 
ж 
Ж 
e 
E 
E 
ne. 
зи 
Ld 
E 


bod 
m 
щй. 
d 
ж 
ame 
m 
жй 
Ed 
ж 
жш. 
ж 


ж. 
в 
эе 
# 
- 
же 


чы 
w 
bd 
m 
di 
E 
Ж 


га 
e 
iud 
E G 
Ж. 
E 
E 


ELIT 


Instrucciones 
del 280 


Introducción 


Antes de analizar una por una todas las instrucciones del 
Z80 y de explicar en detalle su finalidad, el efecto que provocan 
en las banderas (flags) y cómo pueden combinarse con los 
diversos modos de direccionamiento, estudiaremos los tipos de 
aquellas que deben existir en cualquier ordenador de tipo 
general. El capítulo 5 está integramente dedicado a la discusión 
en profundidad de las técnicas de direccionamiento. 


Clases de instrucciones 


Las instrucciones pueden clasificarse con arreglo a muchos 
criterios. En este libro reconoceremos las siguientes cinco cate- 


gorías: 
1. Transferencia de datos. 
2. Tratamiento de datos. 
3. Verificación y bifurcación. 
4. Entrada y salida. 
5. Control. 


Vamos ahora a describirlas una por una. 
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TRANSFERENCIA DE DATOS 


Estas instrucciones llevan datos de unos registros a otros, 
entre los registros y la memoria o entre los registros y algún 
dispositivo de entrada/salida. Puede haber instrucciones de 
transferencia específicas para registros especializados; así, hay 
operaciones de inserción (push) y extracción (pop), para mane- 
jar la pila con más eficacia; cargan una palabra de datos de la 
posición superior de la pila en el acumulador, y actualizan 
automáticamente el registro del apuntador de la misma, todo 
ello en una sola instrucción. | 


TRATAMIENTO DE DATOS 


Las instrucciones de tratamiento de datos pueden, a su vez, 
clasificarse en cinco categorías generales: 


Operaciones aritméticas (como adición o sustracción). 
Manipulación de bits (SET y RESET). 

Incremento y decremento. 

Operaciones lógicas (AND, OR, OR exclusivo). 
Operaciones de desplazamiento y rotación (SHIFT, 
ВОТАТЕ). 


Para manipular los datos con eficacia es muy deseable 
disponer de instrucciones aritméticas potentes, como la multipli- 
cación y división, aunque, por desgracia, están ausentes de la 
mayor parte de los microprocesadores. Son también muy intere- 
santes instrucciones potentes de desplazamiento y rotación, 
como desplazar n bits o intercambiar nibbles (es decir, cambiar 
de posición las mitades izquierda y derecha de un byte), aunque 
también éstas faltan de casi todos los microprocesadores. 

Antes de examinar las instrucciones del Z80, recordemos la 
diferencia entre desplazamiento y rotación. El desplazamiento es 
la traslación del contenido de un registro o de una posición de 
memoria a la izquierda o a la derecha en un bit; como 
resultado sale del registro un bit, que pasa al bit de acarreo. El 
bit que entra por el lado opuesto será “0”, salvo que se haya 
efectuado un “desplazamiento aritmético a la derecha”, que 
tiene como resultado la duplicación del bit más significativo. 

En la rotación, el bit que sale también pasa al acarreo; pero 
el que entra es el que había en dicho acarreo antes del inicio de 
la operación; por tanto, equivale a una permutación circular de 
9 bits. Con frecuencia conviene disponer de una instrucción de 
permutación de 8 bits que traslade el bit de un extremo al 
contrario; no la tiene casi ningún microprocesador, aunque si el 
780 (véase figura 4.1). 


PE EN кє 


Figura 4.1 
Desplazamiento y rotación. 


DESPLAZAMIENTO A LA IZQUIERDA 


ACARREO 


ROTACION A LA IZQUIERDA 


EAS, 
pm 


Por último, al desplazar una palabra a la derecha es conve- 
niente disponer de otro tipo de desplazamiento, llamado exten- 
sión de signo o “desplazamiento aritmético a la derecha”. Al 
operar en complemento a dos, sobre todo al ejecutar rutinas de 
punto flotante, suele ser necesario desplazar un nümero negati- 
vo a la derecha; en este caso, el bit que entra por la izquierda 
ha de ser un “1”, porque el signo debe conservarse tantas veces 
cuantas se efectúe el desplazamiento; esto es lo que se llama 
desplazamiento aritmético a la derecha. 


VERIFICACION Y SALTO 


Las instrucciones de verificación comprueban si los bits 
contenidos en el registro que se especifica son “0”, “1” o alguna 
combinación determinada de esos valores. Como mínimo, es 
necesario que pueda verificarse el registro de estado, lo que 
significa que conviene disponer en ese registro del mayor núme- 
ro posible de banderas. Es aconsejable que la comprobación de 
tales bits pueda hacerse con una sola instrucción. Lo mejor, sin 
duda, sería poder verificar cualquier bit de cualquier registro y 
comparar el valor de cualesquiera otros registros (mayor que, 
menor que, igual a). Las instrucciones de la mayor parte de los 
microprocesadores sólo sirven para verificar bits aislados del 
registro de estado, pero el Z80 está mejor equipado. 

Las instrucciones de salto de las que se suele disponer se 
organizan en tres categorías: 


1. Salto, que especifica una dirección completa de 16 bits. 

2. Salto relativo, normalmente limitado a un campo de 
desplazamiento de 8 bits. 

3. Llamada, que se emplea en combinación con subrutinas. 
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Conviene disponer de saltos de dos o incluso tres salidas 


(mayor que, menor que o igual a, por ejemplo). Es cómodo el 


salto relativo, que es el salto hacia adelante o hacia atrás de 
unas pocas instrucciones, aunque, en realidad, equivale al salto 
normal. Al término de la mayor parte de los bucles hay una 
operación de incremento o decremento seguida por otra de 
verificación y bifurcación; por tanto, la disponibilidad de todas 
estas operaciones en una sola instrucción influye decisivamente 
en la ejecución eficaz de bucles. Lamentablemente, la mayor 
parte de los microprocesadores sólo disponen de bifurcaciones 
sencillas combinadas con verificaciones igualmente sencillas, lo 
que, naturalmente, complica la programación y reduce la efica- 
cia. E] 780 sí dispone de una instrucción de “decremento y 
salto”, aunque sólo comprueba si un registro determinado (B) 
vale 0. 


ENTRADA/SALIDA 


Las instrucciones de entrada/salida sirven para manipular 
los: dispositivos de entrada/salida. La mayoría de los micropro- 
cesadores de 8 bits trabajan con EJS direccionada en memoria; 
esto implica que los dispositivos de entrada y salida están 
conectados al bus de direcciones igual que las pastillas de 
memoria y se direccionan de la misma forma. A efectos de 
programación se tratan como posiciones de memoria que, por 
lo general, necesitan 3 bytes y, en consecuencia, son lentas. Lo 
mejor para aumentar la eficacia es contar con un mecanismo de 
direccionamiento breve para que los dispositivos de E/S, en los 
que la velocidad es crucial, puedan residir en la página 0. Sin 
embargo, si la página O tiene direccionamiento, suele ser, por lo 
general, para la memoria КАМ, lo que impide su utilización 
eficaz para los dispositivos de E/S. El 780, como el 8080, 
dispone de instrucciones especiales de entrada y salida, de 
manera que el programador puede direccionar los dispositivos 
de E/S como posiciones de memoria o como tales, utilizando 
para ello instrucciones E/S, que describiremos más adelante en 
este mismo capítulo. 


INSTRUCCIONES DE CONTROL. 


Las instrucciones de control proporcionan las señales de 
sincronización y pueden suspender o interrumpir un programa. 
También funcionan como paradas o interrupciones simuladas 
(las interrupciones se tratarán en el capítulo 6, dedicado a las 
técnicas de entrada y salida). 


Las instrucciones del 280 


INTRODUCCION 


El microprocesador Z80 se ha construido como sustituto del 
8080, al que supera en algunos aspectos; contiene todas las 
instrucciones de éste y algunas nuevas. Como el código de 
operación de 8 bits ofrece poco espacio, asombra que quienes 
proyectaron el 780 hayan conseguido añadirle más instruccio- 
nes; para ello aprovecharon unos códigos de operación sin usar 
que habia en el 8080 y añadieron un byte nuevo al código de 
las operaciones indexadas. Por eso algunas instrucciones del 
Z80 ocupan hasta 5 bytes de memoria. 

Es importante recordar que un mismo programa puede 
escribirse de muchas formas diferentes. Para programar con 
eficacia es imprescindible conocer a fondo y entender todas las 
instrucciones, aunque cuando se está aprendiendo no hay nece- : 
sidad de escribir programas optimizados. Asi es que, al acercar- 
se por vez primera a este capítulo, lo que importa no es 
recordar las instrucciones en su totalidad, sino estudiar las 
categorías y los ejemplos típicos. Más adelante, en el momento 
de escribir programas, el lector podrá consultar la descripción 
de las instrucciones y escoger las más adecuadas a sus necesida- 
des. En esta sección trataremos de simplificar las instrucciones 
del Z80 y de agruparlas en categorías lógicas. El lector interesa- 
do en explorar las posibilidades de cada una de ellas no tiene 
más que consultar las descripciones individuales. 

Analizaremos las posibilidades del Z80 en términos de las 
cinco categorías de instrucciones definidas al principio de este 
capítulo. 


INSTRUCCIONES DE TRANSFERENCIA DE DATOS 
E] Z80 dispone de cuatro clases de instrucciones dentro de 
esta categoria: para hacer transferencias de 8 bits, para hacer 


transferencias de 16 bits, para realizar operaciones en la pila y 
para transferir bloques. Examinémoslas más despacio. 


TRANSFERENCIAS DE DATOS DE 8 BITS 


Se hacen todas con instrucciones de carga, que tienen el 
formato: 


LD destino, origen 
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Así, para cargar el acumulador А a partir del registro В, se 
utilizaría la instrucción 


LD A,B 


Pueden hacerse transferencias directas entre dos registros de 
trabajo cualesquiera (A, B, C, D, E, H, L). 

Para cargar un registro de trabajo a partir de una posición 
de memoria, excepción hecha del acumulador, hay que cargar la 
dirección de dicha posición en el par de registros HL. 

Por ejemplo, para cargar el registro С con la posición de 
memoria 1234, primero se cargan los registros H y L con el 
valor “1234”, utilizando para ello una instrucción de carga de 
16 bits, que describiremos en la próxima sección. A continua- 
ción se utiliza la instrucción LD C,(HL), que dará lugar al 
resultado que se busca. 

El acumulador constituye una excepción, porque puede 
cargarse directamente a partir de cualquier posición especificada 
de la memoria. Esto se llama modo de direccionamiento exten- 
dido. Por ejemplo, para cargar el acumulador con el contenido 
de la posición de memoria 1234, se emplea la instrucción 
siguiente: 


LD А, (1234H) (Obsérvese el uso de “()” para denotar 
“el contenido de”) 


La instrucción se almacenará en la memoria como sigue: 


dirección PC :3A (código de operación) 
PC+1:34 (byte inferior de la dirección) 
РС--2:12 (byte superior de la direc- 
ción) 


Obsérvese que en la instrucción propiamente dicha la direc- 
ción se almacena en orden inverso: 


Los registros de trabajo pueden también cargarse con cual- 
quier valor especificado de 8 bits o “literal” contenido en el 
segundo byte de la instrucción (esto se llama direccionamiento 
inmediato). Ejemplo: 


LD E, PH 


que carga en el registro E el valor hexadecimal 12, 


Figura 4.2 
Instrucciones de carga de 8 
bits (“LD”). 


En la memoria, la instrucción aparece asi: 


РС :1Е (código de operación) 
PC+1:12 (operando literal) 


Como resultado de esta instrucción aparecerá en el registro 
E el operando inmediato o valor literal. 

También pueden cargarse registros en modo de direcciona- 
miento indexado, como veremos detalladamente en el próximo 
capítulo. Para cargar registros específicos hay otras posibilida- 
des, que recoge la tabla de la figura 4.2; las zonas sombreadas 
muestran las instrucciones comunes al 8080A. 


DESTINO 


00 00 со ор 00 

77 70 n 73 74 

а d а а d 
FO FD FD 
72 73 74 
d 9 4 


TRANSFERENCIAS DE DATOS DE 16 BITS 


En términos generales, cualquier par de registros de 16 bits, 
BC, DE, HL, SP, IX, IY, puede cargarse con un operando 
literal de 16 bits, a partir de una dirección de memoria especifi- 
cada (direccionamiento ampliado) o a partir de la parte superior 
de la pila (es decir, con la dirección contenida en SP). A su vez, 
los contenidos de esos pares de registros pueden almacenarse de 
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Figura 4.3 
Instrucciones de carga de 16 
bits (“LD”, "PUSH" y “POP”). 
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la misma forma en cualquier posición especificada de la memo- 
ria o en la parte superior de la pila. Además, el registro SP 
puede cargarse a partir de HL, IX e IY, lo que facilita la 
creación de varias pilas. También el par de registros AF puede 
introducirse en la parte superior de la pila. 

La tabla de la figura 4.3 recoge todas las posibilidades. Las 
operaciones de extracción e introducción en la pila se conside- 
ran transferencias de datos de 16 bits; dichas operaciones 
transfieren los contenidos de un par de registros a la pila y 
viceversa. Obsérvese que no hay instrucciones únicas de intro- 
ducción y extracción de la pila para guardar registros indivi- 
duales de 8 bits. 


FUENTE 


REGISTRO 


DESTINO 


INSTRUCCIONES —9 
DE INTRODUCCION 


NOTA: Las instrucciones de introducción y extracción INTRUCCIONES 
ajustan el SP tras cada ejecución. DE EXTRACCION 


La introducción o extracción de un doble byte se hace 
siempre sobre un par de registros: AF, BC, DE, HL, IX, IY 
(véanse la fila inferior y la columna derecha de la figura 4.3). 

Para operar con AF, BC, DE y HL basta una instrucción de 
un byte, lo que da lugar a una eficacia considerable. Suponga- 
mos, por ejemplo, que el apuntador de la pila SP contiene el 
valor “0100”. Se ejecuta la instrucción 


PUSH AF 


Figura 4.4 


.. Intercambios "EX" y “EXX”. 


Al introducir en la pila el contenido del par de registros, 
primero se decrementa el apuntador SP, y a continuación se 
deposita en la parte superior de la pila el contenido del registro 
A; acto seguido vuelve a decrementarse SP, y pasa a la pila el 
contenido de F. Al término de la transferencia a la pila, SP 
señala el elemento superior de la misma, que en nuestro ejem- 
plo es el valor de Е. 

Es importante recordar que en el Z80 SP señala la parte 
superior de la pila y se decrementa cada vez que se carga un par 
de registros. Hay procesadores en los que no rige la misma 
convención, lo que puede causar confusiones. 


INSTRUCCIONES DE INTERCAMBIO 


El código mnemónico EX controla las operaciones de in- 
tercambio que, en realidad, son transferencias dobles de datos. 
La instrucción EX modifica los contenidos de dos posiciones 
especificadas. Аз, EX puede usarse para intercambiar la parte 
superior de la pila con HL, IX e IY y también para intercam- 
biar los contenidos de DE y HL y de AF y AF' (recuerde que 
AF' corresponde al otro par de registros AF con que cuenta el 
280). 

Рог último, hay una instrucción especial EXX para inter- 
cambiar los contenidos de BC, DE y HL con los de los 
registros correspondientes del segundo banco del Z80. 

La figura 44 resume todos los intercambios posibles. 


DIRECCIONAMIENTO ая 


| Ar [ec oean | н. | ix | 
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INSTRUCCIONES DE TRANSFERENCIA DE BLOQUES 


Mediante estas instrucciones se transfiere no un byte sencillo 
o doble, sino un bloque completo de datos. Para el fabricante 
son más difíciles de realizar que casi todas las demás instruccio- 
nes, y por eso faltan en la mayor parte de los microprocesado- 
res; pero resultan muy cómodas para el programador y aumen- . 
tan la eficacia de los programas, sobre todo de las operaciones 
de entrada/salida. Sus aplicaciones y sus ventajas quedarán 
claras a lo largo del resto del libro. El Z80 dispone de algunas 
instrucciones automáticas de transferencia de datos que em- 
plean convenciones especificas. 

Todas esas instrucciones exigen el empleo de tres pares de 
registros: BC, DE, HL. 

BC se utiliza como contador de 16 bits, lo que significa 
que pueden transferirse automáticamente hasta 219 = 64K. 
HL actúa como apuntador fuente capaz de señalar cualquier 
posición de la memoria. DE es el apuntador destino y también 
puede señalar cualquier punto de la memoria. 

Hay cuatro instrucciones de transferencia de bloques: 


LDD, LDDR, LDI, LDIR 


Todas ellas decrementan el registro contador BC tras cada 
transferencia. Dos de ellas —LDD y LDDR— decrementan los 
registros apuntadores DE y HL, y las otras dos —LDI y 
LDIR— los incrementan. En los dos grupos de instrucciones, la 
letra R del final del código mnemotécnico significa repetición 
automática. Analicemos estas instrucciones con un poco más de 
detalle. 

LDI significa "carga e incremento"; transfiere un byte desde 
la posición de memoria señalada por H y L hasta el destino 
señalado por D y E, y, además, decrementa BC. Incrementa 
automáticamente H y L, y D y E, de manera que todos los 
pares de registros quedan correctamente dispuestos para llevar 
a cabo la siguiente transferencia de bytes en el momento en que 
sea necesario. 

LDIR significa “carga, incremento y repetición”, y ejecuta 
LDI las veces necesarias para que el registro contador BC 
alcance el valor “0”. Se utiliza para desplazar automáticamente 
un bloque continuo de datos desde un área de memoria a otra. 

LDD y LDDR funcionan de la misma forma, con la diferen- 
cia de que los apuntadores de direcciones se decrementan en 
lugar de incrementarse; por tanto, la transferencia empieza por 
la dirección más alta del bloque en lugar de por la más baja. La 
figura 4.5 resume los resultados de las cuatro instrucciones. 


Figura 4.5 
Grupo de transferencia de blo- 
ques. 


Hay instrucciones automáticas similares de comparación 
(CP) que se resumen en la figura 4.6. 


FUENTE 


‘LDI’ — cargar (DE — (HL) 
Inc Ht & DE, Dec BC 


'LDIR,' — cargar (DE) -«—(HL) 
Ine ns DE, Dec BC, Repetir hasta que 


DESTINO (NOIR | (DE) 


‘LDD’ — cargar (DE) -4— (HL) 
Dec HL & DE, Dec ВС 
rs 


'LDDR' - cargar (DE)-4——(HL) 
Dec HE & DE, Dec BC, Repetir hasta que 


Reg HL apunta a la fuente 
Reg DE apunta al destino 
Reg BC contador de bytes 


INSTRUCCIONES DE TRATAMIENTO DE DATOS 


Aritméticas 


Hay dos operaciones aritméticas básicas: suma y resta, que 
hemos empleado repetidas veces en el capítulo anterior. Hay 
dos tipos de suma, con y sin acarreo, codificadas como ADC y 
ADD, respectivamente. También hay dos instrucciones de resta, 
con y sin acarreo: SBC y SUB. 

A este grupo pertenecen también las tres instrucciones 
especiales DAA, CPL y NEG. La instrucción de ajuste decimal 
del acumulador DAA sirve para realizar operaciones BCD, y nor- 
malmente se usa en todas las sumas y restas ejecutadas en ese có- 
digo. CPL calcula el complemento a uno del acumulador, y 
NEG hace negativo el acumulador en el formato de su comple- 
mento (complemento a dos). 

Todas las instrucciones anteriores operan sobre datos de 
ocho bits. Las operaciones de 16 bits son más limitadas. Como 
describe la figura 4.8, se dispone de ADD, ADC y SBC para 
registros especificos. 

Por ültimo, hay instrucciones de incremento y decremento 
que operan sobre todos los registros, tanto en el formato de 8 
bits como en el de 16. Se resumen en la figura 4.7 (operaciones 
de 8 bits) y en la 4.8 (operaciones de 16 bits). 
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POSICION 
BUSCADA 


"СР! 
Inc HL, Dec ВС 


'CPIR', Inc HL, Dec ВС 
Repetir hasta que BC = 0 
o hasta coincidencia 


‘CPD’ Dec HL & BC 


'CPDR' Dec HL & BC 
Repetir hasta que BC = 0 
o hasta coincidencia 


Figura 4.6 М da de bl HL señala la posición de memoria, que debe 
Grupo de búsque a е о- compararse con el contenido del acumulador 


ques. BC es el contador de bytes 


FUENTE 


DIRECCIONAMIENTO DE REGISTROS 


(пука) 


RESTA con 
ACARREO'SBC' 


a та т сета тоа с п е. т 
dE JS 


B6 
d 
FD 
BE 
d 


Figura 4.7 DECREMENTAR 


Aritmética y lógica de 8 bits. 
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Figura 4.8 


Aritmética y lógica de 16 bits. 


Obsérvese que, en general, todas las operaciones aritméticas 
modifican algunas de las banderas; este aspecto se describe 
detalladamente en la exposición pormenorizada de las instruc- 
ciones contenida más adelante en este mismo capítulo. No 
obstante, hay que subrayar que las instrucciones INC y DEC 
que operan sobre pares de registros no modifican ninguna de 
las banderas; es un detalle importante que conviene tener en 
cuenta; supone que si se incrementa o se decrementa uno de los 
pares de registros hasta el valor “0”, no se fijará a 1 el bit Z del 
registro de banderas F, de forma que es preciso verificar 
explicitamente en el programa si el valor del registro es “0”. 

También conviene señalar que las instrucciones ADC y SBC 
siempre afectan a todas las banderas. Esto no significa que 
necesariamente todas las banderas sean diferentes tras su ejecu- 
ción, sino que pueden serlo. 


FUENTE 


DD Е 
09 


DESTINO 


SUMA CON ACARREO 
Y ACTIVACION ED ED En 
BANDERAS “АРС: 4А бА 
RESTA CON ACARREO 
Y ACTIVACION HL ED 
BANDERAS 'SBC' 72 


INCREMENTO “INC* 


DECREMENTO “DEC” 


Lógicas 


Hay tres operaciones lógicas: AND, OR y XOR (exclusivo), 
y una instrucción de comparación CP. Todas ellas operan 
exclusivamente sobre datos de 8 bits. La tabla de la figu- 
ra 4.7 recoge todas las posibilidades y códigos de operación 
de estas instrucciones, que estudiaremos ahora con cierto de- 
talle, 
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AND 


Cada operación lógica está caracterizada por una tabla de 
verdad, que expresa el valor lógico del resultado en función de 
las entradas. La tabla de verdad de AND es la siguiente: 


La operación AND se caracteriza porque el resultado sólo 
es “1”, si las dos entradas son “1”; en otras palabras, si una de 
las entradas es “0”, el resultado siempre será “0”. Esta caracte- 
rística se emplea para igular a 0 un bit de una palabra, y se 
llama “enmascarar”. 

Una aplicación importante de la instrucción AND es elimi- 
nar o enmascarar uno o más bits especificos de una palabra. 
Supongamos que queremos igular а O los cuatro bits derechos 
de una palabra; para ello utilizaremos el siguiente programa: 


LD A, PALABRA PALABRA CONTIENE 
“10101010” 

AND  11110000B “11110000” ES LA MAS- 
CARA 


Sea PALABRA igual a “10101010”, Una vez ejecutado el 
programa, en el acumulador aparecerá el valor “10100000”. “B” 
sirve para indicar un valor binario. 


Ejercicio 4.1: Redáctese un programa de tres líneas que iguale a 0 
los bits 1 y 6 de PALABRA. 


Ejercicio 4.2: ¿Qué ocurriría si MASCARA = “11111111”? 


OR 


Esta instrucción ejecuta la operación OR, caracterizada por 
la siguiente tabla de verdad: 


=.. о © 
осоо 
NAA 
- о – Ф 
Ки 
ha © 


Si uno de los operandos de la operación lógica OR es “1”, el 
resultado es siempre “1”. La aplicación obvia de la instrucción 
es igualar un bit a “1”. 

Igualemos a “1” los cuatro bits de la derecha de PALA- 
BRA: 


LD A, PALABRA 
OR 00001111B 


Supongamos que PALABRA contiene “10101010”; el valor 
final del acumulador será “10101111”. 


Ejercicio 4.3: ¿Cuál sería el resultado de la instrucción OR 
10101111B? 


Ejercicio 4.4: ¿Cuál sería el resultado de aplicar la operación OR 
al número hexadecimal “FF”? 


XOR 


XOR significa “OR exclusivo”, que se diferencia del OR 
g q 

que acabamos de estudiar en que el resultado es “1” solamente si 

uno, y nada más que uno, de los operandos vale “1”. En efecto, 

la operación OR normal produce como resultado “1” cuando los 

dos operandos valen "1", mientras que la operación exclusiva 

produce “0” en este mismo caso. La tabla de verdad es: 


= о © 


Esta operación sirve para hacer comparaciones. Si dos 
palabras difieren en un bit cualquiera, el resultado del OR 
exclusivo será “1”. Además, en el Z80 la instrucción sirve 
también para complementar una palabra, porque la instrucción 
de complemento sólo actúa sobre el acumulador. Para ello se 
ejecuta XOR con una palabra formada por unos. El programa 
es: 


LD A, PALABRA 
XOR, 11111111B 
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Figura 4.9 
Desplazamiento y rotación. 
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Supongamos que PALABRA vale “10101010”; el valor final 
del registro será “01010101”. Puede comprobarse que es el 
complemento del valor de partida. 

XOR puede utilizarse como “conmutador de bit”. 


Ejercicio 4.5: ¿Cuál sería el resultado de aplicar XOR a un 
registro con el contenido hexadecimal “00”? 


OPERACIONES DE DESALINEAMIENTO 
(DESPLAZAMIENTO Y ROTACION) 


Empecemos por diferenciar entre desplazamiento y rotación, 
operaciones ambas mostradas en la figura 4.9. El desplazamien- 
to es el traslado del contenido de un registro en una posición de 
un bit hacia la izquierda o hacia la derecha. El bit que sale del 
registro pasa al acarreo C, y el que entra es 0, como ya se 
explicó en la sección anterior. 


DESPLAZAMIENTO A LA IZQUIERDA 


ACARREO 


ROTACION A LA IZQUIERDA 


por 


Hay una excepción, llamada desplazamiento aritmético a la 
derecha. Cuando se realizan operaciones con nümeros negativos 
en formato complemento a 2, el bit de la izquierda corresponde 
al signo, y vale *1". Cuando se divide un nümero negativo por 
“2”, desplazándolo a la derecha, debe mantenerse como negati- 
vo, lo que quiere decir que el bit de la izquierda debe seguir 
siendo “1”. Todo esto lo ejecuta automáticamente la instrucción 
SRA de desplazamiento aritmético a la derecha. Con esta 
instrucción, el bit que entra por la izquierda es idéntico al del 
signo: es “0” si ése era su valor, y es “1” si valía “1”. Los 
resultados se recogen en la figura 4.10, que resume todas las 
operaciones de desplazamiento y rotación posibles. 


Rotación 


La rotación se diferencia del desplazamiento en que el bit 
que se incorpora al registro procede bien del extremo opuesto 


TIPO 


DE ROTACION 
о 


DESPLAZAMIENTO: 
E 


Figura 4.10 


Rotaciones y desplazamientos. 


Figura 4.11 
Rotación de 9 bits. 


Figura 4.12 
Rotación de 8 bits. 


Fuente y destino 


T т 


Rotación circular 
El м a la izquierda 


[acd Rotación circular 


a la derecha 


Г] Rotación а la izquierda 


Lap Rotación 

pom a la derecha 
Desplazamiento 

EX aritmético a la izquierda 


— 
-- 
Desplazamiento 
NE aritmético a la derecha 
n— Desplazamiento 
o mH lógico a la derecha 
К Rotación 
ec CEDAT a la izquierda 


Rotación 
a la derecha 
ACC 


de ese mismo registro, bien del acarreo. El Z80 dispone de dos 
tipos de rotación, de 8 y de 9 bits. 

La rotación de 9 bits se muestra en la figura 4.11. Si es 
hacia la derecha, los ocho bits del registro se desplazan uno 
hacia ese lado; el que sale por la derecha pasa al acarreo, como 
es habitual, cuyo valor anterior —antes de ser reemplazado por 
el que acaba de entrar— pasa al extremo izquierdo. En mate- 
máticas se llama a esto rotación de 9 bits, porque supone el 
desplazamiento de los ocho bits del registro más el del acarreo. 
La rotación a la izquierda funciona exactamente igual, pero en 
sentido contrario. 


7 REGISTRO 0 


С 
DOSES Мез q E y 
С 


. La rotación de 8 bits funciona igual. El bit 0 pasa a ocupar 
el lugar del bit 7 o viceversa, según el sentido de la operación. 
Además, el bit que sale del registro se introduce también en el 
acarreo. La secuencia queda mostrada en la figura 4.12. 


A LA DERECHA 


A LA IZQUIERDA 
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Figura 4.13 
Instrucciones de rotación de ci- 
fras (rotación decimal). 
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Instrucciones especiales para cifras 


Hay dos instrucciones especiales de rotación de cifras que 
facilitan la aritmética BCD. El resultado es una rotación de 
cuatro bits entre dos cifras contenidas en la posición de memo- 
ria señalada por los registros HL y la cifra que ocupa la mitad 
inferior del acumulador. El funcionamiento se ilustra en la 
figura 4.13. 


MEMORIA 


MEMORIA 


MANIPULACION DE BITS 


Ya hemos visto que las operaciones lógicas pueden utilizarse 
para fijar o modificar bits o grupos de bits en el acumulador. 
No obstante, resulta cómodo poder fijar o modificar cualquier 
bit de cualquier registro o de cualquier posición de la memoria 
con una sola instrucción. Es una opción que exige un número 
considerable de códigos de operación y que, por ello, falta en la 
mayor parte de los microprocesadores. Sin embargo, el Z80 
cuenta con numerosas posibilidades de manipulación de bits, 
que se resumen en la figura 4.14; la tabla recoge también las 
instrucciones de comprobación, que describiremos en la sección 
siguiente. 

Hay dos instrucciones especiales para operar en la bandera 
de acarreo: CCF (complementar bandera de acarreo) y SCF 
(poner la bandera de acarreo); las dos aparecen en la figura 
4.15. 

Nota: Se suele usar OR А o bien AND А para quitar la 
bandera de acarreo con una operación de un solo byte. 
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ecd afe en 
803/8808 ЕЕ 88.588 
запас 
л. 
© 

Inmensae 
g 

ЗО сиесиоси 


RECCIONAMIEN 


000 


.14 


4 
Instrucciones de manipulación 


de bits. 


е 

Le 

3 
S 
ir 


Figura 4.15 
Operaciones con AF de tipo 
general. 


Figura 4.16 
Registro de estado. 
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VERIFICACION Y SALTO 


Dado que las operaciones de verificación dependen en gran 
medida de la manipulación del registro de estado, empezaremos 
por describir la función de cada una de las banderas. El 
contenido del registro puede estudiarse en la figura 4.16. 


Ajuste decimal del 
acumulador, “DAA” 


L 


Complementar acumulador 
"CPL" 


Negativizar acumulador, “МЕС” 
(complemento a dos) 


Complementar bandera de 
acarreo “ССЕ” 


Poner bandera de acarreo, 
"все" 


7 6 5 4 3 2 1 о 
БИРЕР ЕЗ 
(0 (0 (T) 


(т) 


С es el acarreo; М denota suma o resta; P/V, paridad o 
desbordamiento; H es el acarreo de la mitad; Z es cero, y 5 es 
signo. Los bits 3 y 5 (“—”) del registro de estado no se utilizan. 
Las dos banderas H y N se emplean en aritmética BCD y no 
pueden verificarse. Las otras cuatro (C, P/V, Z y S) pueden 
verificarse en combinación con instrucciones condicionales de 
salto o llamada. La función de cada una de las banderas se 
describirá en los próximos párrafos. 


Acarreo (C) 


En casi todos los microprocesadores, y en el Z80 en particu- 
lar, el bit de acarreo desempeña una función doble: en primer 
lugar, indica si una operación de suma o resta ha dado lugar a 
un acarreo; en segundo lugar, es el noveno bit de las instruccio- 
nes de desplazamiento y rotación. La reunión en un único bit 
de las dos funciones facilita algunas ,operaciones, como la 
multiplicación; esto ya debió quedar claro en la sección del 
capitulo anterior dedicada a dicha operación. 


Al estudiar el empleo del bit de acarreo es importante 
recordar que todas las operaciones aritméticas lo igualan a 1 o a 0, 
dependiendo del resultado de las instrucciones. Lo mismo hacen 
.. todas las operaciones de desplazamiento y rotación, dependien- 
do el resultado del valor del bit que sale del registro. 

Las instrucciones lógicas (AND, OR, XOR) siempre ponen a 
0 el bit de acarreo, y pueden emplearse explícitamente para ello. 

Las siguientes instrucciones afectan al bit de acarreo; 
ADD A,s; ADC A,s; SUB $; SBC Аз; CP s; МЕС; AND $; 
ОК s; ХОК s; ADD DDss; ADC HL,ss; SBC HL,;s; RLA; 
RLCA; RRA; RRCA; RL m; RLC m; RR m; RRC m; SLA m; 
SRA m; SRL m; DAA; SCF; CCF. 


Resta (N) 


Normalmente, esta bandera no la usa el programador, sino 
el propio Z80 durante operaciones BCD. El lector recordará, 
sin duda, del capítulo anterior, que tras una suma o una resta 
BCD se ejecutaba una instrucción DAA (ajuste decimal del 
acumulador) para obtener resultados BCD válidos. Pero la 
operación de ajuste tras una suma es diferente que tras una 
resta, lo que quiere decir que la forma en que se ejecute la 
instrucción DAA depende del valor de la bandera N. Esta vale 
“0” tras una suma, y “1” tras una resta. 

El símbolo *N" utilizado para esta bandera puede confundir 
a los programadores acostumbrados a otros microprocesadores, 
que quizá lo tomen erróneamente por el bit de signo. Es un bit 
interno de signo de operación. 

Ponen Ма “0” las instrucciones: ADD A,s; ADC As; AND s; 
OR $; XOR s; INC s; ADD DDss; ADC HLss; RLA; 
RLCA; RRA; RRCA; RL m; ВГС m; RR m; ВВС m; 
SLA m; SRA m; SRL m; RLD; RRD; SCF; CCF; IN г, (C); 
LDI; LDD; ІШІК; LDDR; LD A, Г; LD A, В; BIT b,+s. 

Ponen N a “1” las instrucciones: SUB s; SBC A,s; CP s; 
NEG; DEC m; SBC HL, ss; CPL; INI; IND; OUTI; OUTD; 
INIR; INDR; OTIR; OTDR; CPI; CPIR; CPD; CPDR. 


Paridad/Desbordamiento (P/V) 


La bandera de paridad/desbordamiento desempeña dos fun- 
ciones diferentes. Mediante instrucciones específicas, la bandera 
se iguala a 1 o a 0 en función de la paridad del resultado, que 
se determina contando el número de unos del mismo; si este 
número es impar, el bit de paridad se hace igual a “0” (paridad 
impar); si es par, se iguala a “1” (paridad par). Donde más se 
utiliza la paridad es en los bloques de caracteres (por lo general, 
en formato ASCII). El bit de paridad es un bit adicional que se 
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añade al código de siete que representa el carácter, con el fin de 
garantizar la integridad de los datos almacenados en un dispo- 
sitivo de memoria. De esta forma, si, por ejemplo, uno de los 
bits del código que representa el carácter cambia accidentalmen- 
te a causa de un fallo del dispositivo de memoria (un disco o 
RAM) o por un error de transmisión, cambiará también el 
número total de unos del código, discrepancia que se detecta 
verificando la bandera del bit de paridad. Dicha bandera se 
emplea, en particular, en instrucciones lógicas y de rotación. 
Como es natural, la misma bandera indicará la paridad del 
dato que se está leyendo durante las operaciones de entrada a 
partir de un dispositivo de E/S. 

El lector familiarizado con el 8080 deberá tener en cuenta 
que en éste la bandera de paridad se usa exclusivamente como 
tal. En el Z80 desempeña algunas otras funciones adicionales; 
debe, pues, manejarse con atención al pasar de un microproce- 
sador al otro. 

La segunda aplicación importante de esta bandera en el Z80 
es el desbordamiento (no disponible en el 8080). La bandera de 
desbordamiento ya la estudiamos en el capítulo 1 al exponer la 
notación en complemento a dos. Detecta si, durante la suma o 
la resta, cambia “accidentalmente” el signo del resultado debido 
al desbordamiento del mismo en el bit de signo (recuérdese que, 
en formato de 8 bits, el mayor positivo posible es + 127 y el 
menor negátivo — 128, siempre en complemento a dos). 

Esta bandera realiza en el Z80 otras dos funciones que nada 
tienen que ver con las que acabamos de examinar. 

Al ejecutar instrucciones de transferencia de bloques (LDD, 
LDDR, LDL, LDIR) y de búsqueda (CPD, CPDR, CPI, CPIR), 
la bandera se emplea para comprobar si el registro contador B 
ha alcanzado el valor “0”. Con instrucciones que decrementan, 
la bandera se pone a “0” si el par de registros del contador de 
byte BC es “0”. Con instrucciones que incrementan se modifica 
si ВС — 1 = 0 al principio de la instrucción, es decir, si la 
instrucción reduce BC hasta “0”. 

Por último, al ejecutar las dos instrucciones especiales 
LD A.I y LD A,R, la bandera P/V refleja el valor del biestable de 
validación de interrupciones (IFF2); tal característica puede 
aprovecharse para conservar o verificar este valor. 

Afectan a la bandera P: AND s; ОК $; XOR $; RL m; ВЕС m; 
RR m; RRC m; SLA m; SRA m; SRL m; RLD; RRD; DAA; 
IN r,(C). 

Afectan a la bandera V: ADD A,s; ADC A,s; SUB s; SBC Аз; 
CP s; NEG; INC s; DEC m; ADC HL,ss; SBC HLoss. 

También utilizan esta bandera LDIR; LDDR (puesta a “0”); 
LDI; LDD; CPI; CPIR; CPD; CPDR. 


Bandera de acarreo mitad (H) 


Esta bandera revela el posible acarreo del bit 3 en el bit 4 
durante una operación aritmética. En otras palabras, representa 
el acarreo del nibble de orden inferior en el de orden superior. 
Como es fácil deducir, se emplea, sobre todo, en operaciones 
BCD, y más concretamente, la utiliza el microprocesador para 
el ajuste decimal (DAA) necesario, con el fin de obtener como 
resultado un valor correcto. 

La bandera se pone a 1 cada vez que se ejecuta una suma 
con acarreo del bit 3 al bit 4, y se vuelve al estado inicial si no 
hay tal acarreo. Y viceversa: en una resta se pone a 1 si hay 
acarreo del bit 4 al 3, y se vuelve al estado inicial en caso 
contrario. 

. Influyen en la bandera, la suma, la resta, el incremento, el 
decremento, las comparaciones y las operaciones lógicas. 

Le afectan las instrucciones siguientes: ADD A,r; ADC Ass; 
SUB $; SBC А; СР $; NEG; AND s; OR s; ХОК s; INC s; 
DEC m; RLA; ВЕСА; RRA; RRCA; RL m; RLC m; ВК m; 
ВКС m; SLA m; SR m; SRL m; RED; RRD; DAA; CPL; 
SCF; IN r,(C); LDI; LLD; LDIR; LDDR; LD A; LD AR; 
ВІТ Ыг; CPI; CPIR; СРО; СРОК. 

Obsérvese que el bit H se ve afectado aleatoriamente por las 
instrucciones de suma y resta de 16 bits y por las de entrada y 
salida de bloques. 


Cero (Z) 


La bandera Z se utiliza para verificar si es cero el valor de 
un byte que se ha calculado o que se está transfiriendo. 
También se usa en instrucciones de comparación, para señalar 
una coincidencia, y en algunas otras funciones de diversa índole, 

Si se produce una operación con resultado cero о si se 
transfiere un dato y el byte vale cero, el bit Z se fija a “1”; en 
caso contrario, se lleva al valor inicial “0”. 

En instrucciones de comparación, Z vale “1”, si el resultado 
de la comparación es afirmativo, y “0”, en caso contrario. 

En el 780, esta bandera desempeña otras tres funciones: en 
conjunción con la instrucción BIT, se usa para indicar el valor 
del bit que se quiere comprobar; vale “1”, si dicho bit es “0”, y 
pasa al valor nulo en caso contrario. 

Al utilizar instrucciones especiales de entrada y salida de 
bloques (INI, IND, OUTI, OUTD), la bandera Z vale “1”, si 
D —1=0, y “0” en caso contrario; también vale "1" я el 
contador de byte desminuye hasta “0” (INIR, INDR, OTIR, 
OTDR). 
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Por último, cuando se usan las instrucciones especiales 
IN r{C), Z se pone a “1” para indicar que el byte de entrada 
vale “0”. 

El valor de Z depende, por tanto, de las siguientes instruc- 
ciones: ADD A,s; ADC As; SUB $; SBC Аз; CP s; NEG; 
AND s; OR s; XOR в; INC s; DEC m; ADC HL,ss; SBC НІ; 
RL m; ВГС m; RR m; ВВС m; SLA m; SRA m; SRL m; 
RLD; RRD; DAA; IN r, (C); INI; IND; OUTI; OUTD; 
INIR; INDR; OTIR; OTDR; CPI; CPIR; CPD; CPDR; 
LD АГ; LD AR; BIT bs; NEG. 

Son instrucciones habituales que no modifican el bit Z: 
ADD 00,85; RLA; RLCA; RRA; RRCA; CPL; SCF; CCF; 
LDI; LDD; LDIR; LDDR; INC DD; DEC DD. 


Signo (S) 

Esta bandera refleja el valor del bit más significatvo de un 
resultado o de un byte transferido (bit séptimo). En notación de 
complemento a 2, el bit más significativo representa el signo 
“0”, si el número es positivo, y “1”, sí es negativo. El bit 
número siete se llama también, por ello, bit de signo. 

En la mayor parte de los microprocesadores, el bit de signo 
representa un importante papel en la comunicación con los 
dispositivos de entrada/salida. Como casi ninguno dispone de 
instrucción BIT de verificación de bits específicos de los regis- 
tros o de la memoria, el bit de signo suele ser el más fácil de 
comprobar. Si se desea examinar la situación de un dispositivo 
de entrada/salida, la lectura del registro de estado supone el 
condicionamiento automático del bit de signo, que toma el 
valor del bit siete del registro de estado y puede a continuación 
verificarse fácilmente por medio del programa. Esto explica por 
qué el registro de estado de la mayor parte de las pastillas de 
entrada/salida conectadas a sistemas microprocesadores tienen 
el indicador más importante (por lo general, dispuesto/no dis- 
puesto) en la posición siete. 

El 780 dispone de una instrucción especial BIT. Sin embar- 
go, para verificar una posición de memoria (que puede ser la 
dirección de un registro de estado de E/S) es preciso cargar 
primero la dirección en los registros IX, IY o HL. No existe 
ninguna instrucción para verificar directamente una dirección 
especificada de la memoria (es decir, que esta instrucción no 
tiene direccionamiento directo). Por tanto, el valor de una 
bandera de dispuesto correspondiente a un dispositivo de entra- 
da/salida, y situada en la posición siete, es útil también en el 
780. 

La bandera de signo también la utiliza la instrucción espe- 
cial IN(C) para indicar el signo del dato que está leyendo. 


El bit de signo se ve afectado por las instrucciones: ADD Ass; 
SUB s; SBC A,s; CP s; NEG; AND s; OR s; XOR s; INC s; 
DEC m; ADC HLss; SBC HL,ss; RL m; ВГС m; RR m; 
RRC m; SLA m; SRA m; SRL m; RLD; RRD; DAA; 
IN г, (С); CPI; CPIR; СРО; СРОК; LD А, 1; LD A,r; NEG; 
ADC Ass. 


Resumen de banderas 


Los bits de banderas se emplean para detectar automática- 
mente las situaciones especiales que se producen en el interior 
de la ALU del microprocesador. Como se detectan fácilmente por 
medio de instrucciones especiales, es sencillo emprender accio- 
nes específicas en respuesta a la situación detectada. Importa 
entender bien la función de todos los indicadores disponibles, 
porque la mayor parte de las decisiones internas del programa 
se adoptan en función de los mismos. Así, todos los saltos - 
efectuados dentro del programa terminarán en una u otra 
posición, dependiendo de lo indicado por las banderas. La 
única excepción la constituyen las interrupciones, que describi- 
remos en el capítulo de entrada y salida, y que pueden determi- 
nar el salto a una posición determinada siempre que en las 
patillas del Z80 se reciba una señal procedente del soporte 
físico. 

De momento, basta con recordar la función principal de 
cada uno de los bits estudiados. Al escribir programas, el lector 
podrá consultar las descripciones detalladas de las instrucciones 
que figuran en este mismo capítulo, para verificar el efecto que 
cada una de ellas ejerce sobre las banderas. Normalmente 
pueden ignorarse casi todas, y quien todavía no esté familiariza- 
do con ellas no debe, pues, dejarse intimidar por su aparente 
complejidad. Su funcionamiento irá quedando claro conforme 
vayamos estudiando nuevas aplicaciones. 

La figura 4.17 contiene un resumen de las seis banderas y de 
la forma en que pasan a valer “1” ó “0” en virtud de las 
diferentes instrucciones. 


Instrucciones de salto 


En una bifurcación, el programa se desvía necesariamente a 
una dirección especificada. Es, pues, un punto en el que la 
ejecución secuencial se interrumpe para dar paso a un segmento 
distinto del programa. Los saltos pueden ser condicionales o 
incondicionales. En un salto incondicional se produce una 
bifurcación a una dirección determinada, con independencia de 
cualquier otra condición. 
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Figura 4.17 
Resumen del 


funcionamiento 


de las banderas. 
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INSTRUCCIONES EBD OBSERVACIONES 


ADO A, s; ADC А, s 
SUB s; SBC A, s, CP s, NEG 


AND; 

OR s; XOR + 

INC з 

DEC m 

ADO DD. s 

ADC HL, ss 

SBC HL, ss 

RLA; В1 СА, RRA, RRCA 

RL m; ВЕС m; АЯ m; RRC т 
SLA т; SRA m; SRL m 

RLO, RRO 

DAA 

CPL 

SCF 

ccr 

IN г. (C) 

INI; IND; OUTI; OUTO 

INIR; INOR; опа: OTOR 

LOI, LOO 

LOIR, LOOR 


CPI, CPIR, CPO, CPOR 


LOA 1; ОА. Я 


ВІТЬ, 


МЕС 


Notación de la tabla: 


-----е 0.00 


v*cceoccoo 


© —x x v» € € o v» v 


oo-oo-ooo 
сохжжх--о- 


20--000-.o 
ооххохо - ~ о 


x 


Suma de 8 bits o suma con acarreo 

Resta de 8 bits, resta con acarreo, 
comparación y negación del acu- 
mulador 

Operaciones lógicas 

Y activación diversas banderas 

Incremento de 8 bits 

Decremento de 8 bits 

Suma de 16 bits 

Suma de 16 bits con acarreo 

Resta de 16 bits con acarreo 

Rotación acumulador 

Rotación y desplazamiento posición m 


Rotación de digitos a izad. y der. 
Ajuste decimal del acumulador 
Complementar el acumulador 

Poner acarreo 

Complementar acarreo 

Entrada a registro indirecta 

Entrada y salida de bloques 

2-0 яВ + 0: en caso contrario, 2 = 1 


Instrucciones de transferencia de blo- 


ques P/V = 1 si BC #0; en caso à 


contrario, P/V = 0. 
Instrucciones de búsqueda de bloques 
Z-1s A = (HL 
en caso contrario, 2 = 0 
P/V = 1 si BC #0, 
en caso contrario, P/V = 0 
El contenido del biestable de inte- 
rrupciones (IFF) se copia en la ban- 
dera P/V 
El complemento del bit b de la 
posición se copia en la bande- 
ra 2 
Negación del acumulador 


SIMBOLO OPERACION 

с Bandera acarreo/unión. С - 1 si la operación produce acarreo del bit más sig- 
nificativo del operando o el resultado. 

2 Bandera сего. Z = 1 si el resultado de la operación es сего. 

5 Bandera de signo. 5 = 1 si el BMS del resultado es uno. 

P/V Bandera de paridad o desbordamiento. Paridad (P) y desbordamiento (V) 
comparten la misma bandera. Las operaciones lógicas afectan a esta bandera 
con la paridad del resultado, mientras que las aritméticas la afectan con el 
desbordamiento del resultado. Si P/V responde a la paridad, P/V = 1 si el re- 
sultado de la operación es par, y P/V = 0, si es impar. Si la bandera responde 
al desbordamiento, P/V = 1 si el resultado de la operación produce desborda- 
miento. 

H Bandera de semiacarreo. H = 1 si las operaciones de suma o resta producen 
acarreo hacia o desde el bit 4 del acumulador. 

N Bandera de suma/resta. N = 1 si la operación anterior fue una resta. 

Las banderas H y М se usan en combinación con la instrucción de ajuste deci- 
mal (DAA) para corregir el resultado en formato BCD empaquetado tras una 
suma o una resta con operandos en formato BDC empaquetado. 

t La posición de la bandera depende del resultado de la operación. 

е La operación no modifica la bandera. 

о La operación pone la bandera а 0. 

1 La operación pone la bandera a 1. 

х No hay que preocuparse por la bandera. 

У La bandera P/V depende del desbordamiento del resultado de la operación. 

P La bandera P/V depende de la paridad del resultado de la operación. 

r Cualquiera de los registros de la UCP: А, B, C D, E, H, L 

5 Cualquier posición de 8 bits para todos los modos de direccionamiento сот- 
patibles con la instrucción de que se trate. 

ss Cualquier posición de 16 bits para todos los modos de direccionamiento com- 
patibles con la instrucción de que se trate. 

ii Cualquiera de los dos registros de índice IX o IY. 

R Contador de refresco. 

n Valor de 8 bits dentro del intervalo <0, 255 >. 

nn Valor de 16 bits dentro del intervalo <0, 65535 >. 

m Cualquier posición de 8 bits para todos los modos de direccionamiento com- 


patibles con la instrucción de que se trate. 


Por cortesía de Zilog, Inc 


En un salto condicional la secuencia de ejecución pasa a una 
dirección especificada sólo si se cumplen una o más condiciones. 
Es el tipo de instrucción que se emplea siempre que hay que 
tomar decisiones basadas en el valor de los datos o de los 
resultados calculados, 

Para poder estudiar las instrucciones de salto condicionales 
es imprescindible entender la función del registro de estado, 
porque todas las decisiones de bifurcación se basan en las 
banderas. Dado que ya las estudiamos en la sección anterior, 
pasaremos ahora a analizar las instrucciones de salto de que 
dispone el 780. 

El microprocesdor cuenta con dos categorías de salto: el 
salto dentro del mismo programa y el salto a una subrutina 
(CALL) y la vuelta de la misma (RETURN). Tras cualquier 
instrucción de salto, el contador del programa PC se carga con 
una nueva dirección, a partir de la que se reanuda la ejecución 
del programa. Las posibilidades de las instrucciones de salto 
sólo pueden captarse plenamente en el contexto de los diversos 
modos de direccionamiento con que cuenta el microprocesador, 
por lo que dejaremos pendiente este aspecto de la discusión 
hasta el siguiente capítulo, que tratará de las técnicas de 
direccionamiento, y nos limitaremos aquí a estudiar otras carac- 
terísticas de tales instrucciones. 

Como ya se ha dicho, los saltos pueden ser incondicionales 
(ramificación a una dirección de memoria especificada) o condi- 
cionales. En este caso, puede comprobarse una de las banderas 
7, С, P/V o S para averiguar si vale “0” o “1”. 

Las abreviaturas correspondientes son: 


Z = сего (Z = 1) 
NZ = no сего (Z - 0) 

С = acarreo (С = 1) 
МС = по acarreo (С = 0) 
РО = paridad impar 
РЕ = paridad par 

Р = positivo (5 = 0) 

М = negativo (S = 1) 


Además, el 780 dispone de una instrucción combinada 
especial que decrementa el registro B y salta a una dirección de 
memoria especificada, siempre que no sea cero. Es una instruc- 
ción potente que se emplea para terminar bucles y que ya 
hemos tenido ocasión de poner a prueba en el capitulo ante- 
rior; su código simbólico es DJNZ. 

También las instrucciones CALL y RET (retorno) pueden 
ser condicionales e incondicionales. Comprueban las mismas 
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Figura 4.18 


Instrucciones de salto. 
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banderas de las instrucciones de bifurcación que acabamos de 
ver. 

Las de bifurcación condicional son instrucciones potentes; 
por lo general, faltan en otros microprocesadores de 8 bits. 
Mejoran la eficacia del programa, porque si no existiesen harían 
falta dos instrucciones para hacer lo mismo. 

Hay dos instrucciones de retorno especiales reservadas a 
rutinas de interrupción, llamadas RETI y RETN, que estudiare- 
mos en la sección de interrupciones del capítulo 6. 

Los modos de direccionamiento y los códigos de operación 
de las bifurcaciones se encuentran en la figura 4.18. 


CONDICION 


PARI- | PARI- 
IN- DAD | ОАО REG. 
COND. AREO RREO CERO PAR IMPAR B40 


RETORNO 
ENEE 
RETORNO DE 
INT. “RETI” (Pen) 
RETORNO DE INT.| Ес 
NO ENMASCA- . 
RABLE “RETN” | МОА. (SP) 


En el capítulo 5 se discutirán detenidamente los modos de 
direccionamiento. Al examinar la figura 4.18 se observa que 
muchos de ellos están restringidos. Así, el salto absoluto JP nn 
puede comprobar cuatro banderas, pero sólo dos el salto JR. 

Respecto a estas dos últimas instrucciones, conviene obser- 
var que, aunque JR suele emplearse más que JP, porque ocupa 
un byte menos y facilita la relocalización del programa, no son 
intercambiables (JR no puede verificar las banderas de paridad 
y signo). 

La instrucción de reinicio, RST, es una bifurcación especial 
de un byte que permite saltar a cualquiera de las ocho direccio- 
nes iniciales del extremo inferior de la memoria (que son, en 
notación decimal, 0, 8, 16, 24, 32, 40, 48 y 56). Es una instruc- 


Figura 4.19 
Instrucción de reanudación. 


ción potente, porque sólo ocupa un byte y proporciona una 
bifurcación rápida, por lo que se emplea, sobre todo, para 
responder a interrupciones. No obstante, el programador puede 
emplearla para cualquier otra cosa. La figura 4.19 recoge los 
códigos de operación de esta instrucción. 


“RSTO' 
L ; Я 
L RST 8 
A 
M 
A ; $ 
D RST 16 
A 
A ‘RST 24' 
L 
A 

'RST 32' 
D 
Г 
R р Е 
E RST 40 
с 
с 
1 ‘RST 48" 
[e] 
N 

"RST 56' 


H denota un nümero hexadecimal 


Instrucciones de entrada/salida 


Las técnicas de entrada y salida se describirán pormenoriza- 
damente en el capítulo 6. Baste decir aquí que los dispositivos 
de E/S pueden direccionarse de dos modos: como posiciones de 
memoria, utilizando cualquiera de las instrucciones que ya se 
han descrito para ello, o mediante instrucciones específicas de 
entrada/salida. Las instrucciones normales de direccionamiento 
de la memoria requieren tres bytes, uno para el código de 
operación y dos para la dirección, y otros tantos accesos a la 
memoria; son, por tanto, lentas. Las instrucciones de E/S 
especializadas son más breves y rápidas, pero tienen dos incon- 
venientes. 

En primer lugar, “desperdician” varios de los preciosos y 
escasos códigos de operación disponibles (en efecto, en un 
microprocesador suelen utilizarse sólo 8 bits para formar todos 
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los códigos de operación necesarios). En segundo lugar, exigen 
la emisión de una о más señales especializadas de entrada/sali- 
da; por tanto, también “desperdician” una o más de las pocas 
patillas de que dispone el microprocesador, casi siempre limita- 
das a 40. Debido a estos inconvenientes, los procesadores suelen 
carecer de instrucciones específicas de entrada/salida; sin embar- 
go, dispone de ellas el 8080 (el primer microprocesdor de 8 bits 
de tipo general potente) y el Z80, que, como sabemos, es 
compatible con el 8080, 

La ventaja de las instrucciones de entrada/salida es que son 
más rápidas, porque sólo ocupan dos bytes; no obstante, puede 
conseguirse un resultado similar con un modo de direcciona- 
miento especial llamado direccionamiento de “página 0”, que 
limita la dirección a un campo de 8 bits. Es la solución 
habitualmente elegida en otros microprocesadores. 

Las dos instrucciones básicas de entrada/salida son IN y 
OUT, que transfieren el contenido de las posiciones de E/S 
especificadas a cualquiera de los registros de trabajo, o vicever- 
sa. Normalmente ocupan dos bytes: el primero, reservado para 
el código de operación, y el segundo, para la parte inferior de la 
dirección. El acumulador se emplea para entregar la parte 
superior de la misma, lo que permite seleccionar uno de los 
dispositivos de 64K. No obstante, esto exige cargar cada vez el 
acumulador con el contenido adecuado, lo que puede reducir la 
velocidad de ejecución. 

Además, el 780 dispone de un modo de registro indirecto 
más cuatro instrucciones especializadas de transferencia de 
bloques para entrada y salida. 

En modo de entrada a registro, cuyo formato es IN г. (C), el 
par de registros B y C se usa como apuntador del dispositivo 
de E/S. El contenido de B se deja en la parte superior del bus de 
direcciones, y a continuación se carga el contenido del dispositi- 
vo de E/S especificado en el registro designado por r. 

Lo mismo cabe decir de la instrucción. OUT. 

Las cuatro instrucciones de transferencia de bloques a la 
entrada son: INI, INIR (INI repetida) IND e INDR (IND 
repetida). Las correspondientes para la salida son: OUTI, 
OTIR, OUTD y OTDR. 

En esta transferencia automática de bloques, el par de 
registros H y L se utiliza como apuntador de destino, y el 
registro C, como selector de dispositivo de E/S (uno entre 256 
dispositivos). En las instrucciones de salida, H y L apuntan a la 
fuente. El registro B se usa como contador, y puede incremen- 
tarse o decrementarse; las instrucciones de entrada correspon- 
dientes son INI para incrementar e IND para decrementar. 

INI es una instrucción de transferencia automática de un 


Figura 4.20 


Instrucciones de salida. 


byte. El registro C selecciona el dispositivo de entrada. En este 
dispositivo se lee un byte y se transfiere a la dirección de 
memoria apuntada por H y L, que a continuación se incremen- 
tan en 1; por su parte, el contador В se decrementa en 1. 

INIR es la misma instrucción automatizada; se ejecuta una 
y otra vez hasta que el contador se decrementa hasta “0”. De 
esta forma pueden transferirse automáticamente hasta 256 by- 
tes. Obsérvese que, para conseguir la transferencia total de 256 
bytes exactamente, el registro B debe fijarse al valor “0” antes 
de ejecutar esta instrucción. 

Los códigos de operación de las instrucciones de entrada y 
salida se resumen en las figuras 4.20 y 4.21. 


FUENTE 


REGISTRO E 


SALIDA “OUTI” 
Inc HL, Dec. B 


SALIDA "OTIR”, Inc. HL, 
Dec. В, REPITE SI В + 0 


SALIDA “OUTD” 

Dec HL& B 

SALIDA “ОТЫН”, Dec. © 
HL y B, REPITE SI В +0 


DIRES рн 
DE LA PUERTA 
DE DESTINO 


ORDENES 
DE SALIDA 
DE BLOQUES 


INSTRUCCIONES DE CONTROL 


Estas instrucciones modifican la situación operativa de la 
CPU o manipulan la información de su estado interno. Son 
siete. 

NOP es una instrucción de no operación, que mantiene el 
procesador inactivo durante un ciclo. Acostumbra a utilizarse 
para introducir un retraso deliberado (4 estados — 2 microse- 
gundos con un reloj de 2 MHz) o para cubrir los huecos 
formados en un programa durante la fase de corrección. Para 
facilitar ésta, el código de operación de NOP suele estar 
formado íntegramente por ceros, porque en el momento de la 
ejecución la memoria suele limpiarse, es decir, suele reducirse a 
ceros; la ejecución de NOP no provoca ningún daño y no 
interrumpe la ejecución del programa. 
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Figura 4.21 
Instrucciones de entrada. 
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DIRECCION 
DE LA PUERTA FUENTE 


DESTINO 
DE ENTRADA 


VINI”: ENTRADA 
inc. HL Dec. В 


OS ORDENES DE ENTRADA 
DE BLOQUES 


“IND”: ENTRADA, Dec. HL 
Dec. 8 


“INOR”: ENTRADA, Dec. 
ML Dec. В. REPETÍA SI B/0 


HALT se emplea combinada con interrupciones o con un 
reinicio. Lo que hace es interrumpir el funcionamiento de la 
CPU; ésta reanudará el funcionamiento en cuanto reciba una 
señal de interrupción o de reinicio. En este modo, la CPU 
ejecuta continuamente instrucciones NOP. Durante la fase de 
corrección suele colocarse un alto al final del programa, porque 
habitualmente el programa principal no tiene que hacer ningu- 
na otra cosa. Hecho esto, es necesario volver a ponerlo en 
funcionamiento explícitamente. 

Hay dos instrucciones especiales para invalidar y validar la 
bandera interna de interrupciones: El y DI. Describiremos las 
interrupciones en el capítulo 6. La bandera de interrupciones 
sirve para autorizar o desautorizar la interrupción de un pro- 
grama. Para evitar interrupciones durante una porción específi- 
ca del programa, puede invalidarse mediante esta instrucción 
el biestable o bandera de interrupciones, como veremos en el 
capítulo 6. La figura 4.22 recoge estas instrucciones. 

Por último, el Z80 cuenta con tres modos de interrupción 
(frente а sólo uno en el 8080). El modo 0 corresponde al único 
del 8080; el 1 es una llamada a la posición 038H, y el 2 es una 
llamada indirecta que utiliza el contenido del registro especial 1 
más 8 bits que proporciona el dispositivo interruptor como 
puntero de la posición de memoria cuyo contenido correspon- 
de a la dirección de la rutina de interrupción. Explicaremos es- 
tos modos en el capítulo 6. 


Figura 4.22 
Instrucciones de control de la 
CPU. 


Resumen 


"HALT* 


INVALIDA 
INT. "(DI)" 


VALIDA INT “(El)” 


MODO DE INT. 
о “IMO” MODO 8080A 


MODO DE INT. 
1 "IM1" LLAMADA A LA POSICION 0038, 


MODO DE INT. LLAMADA INDIRECTA USANDO ЕЁ REGISTRO 
2 "IM2" ру 8 BITS DEL DISPOSITIVO INTERRUPTOR 
COMO PUNTERO 


E] Z80 puede recibir dos tipos de interrupciones, que llegan 
por las patillas IRQ y NMI, y que también analizaremos en el 
capítulo 6. 


Ya hemos visto las cinco categorías de instrucciones de que 
dispone el Z80. La sección siguiente recoge los detalles particu- 
lares de cada una de ellas. Para empezar a programar no es 
preciso entender la función de todas las instrucciones, sino que 
basta con conocer unas pocas esenciales. No obstante, quien 
desee escribir buenos programas sí tendrá que estudiarlas y 
conocerlas todas. Como es natural, al principio la eficacia tiene 
poca importancia, y por eso pueden ignorarse la mayoría de las 
instrucciones. 

Todavía no hemos descrito un aspecto muy importante: las 
técnicas de direccionamiento del 780, que facilitan la recupera- 
ción de datos archivados en el espacio de la memoria. Estudia- 
remos dichas técnicas en el capítulo siguiente. 
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INSTRUCCIONES DEL Z80: 
DESCRIPCION INDIVIDUAL 


ABREVIATURAS 
BANDERA ON OFF | 
Acarreo C (acarreo) NC (no acarreo) 
Signo M (menos) P (más) 
Cero Z (cero) NZ (no cero) 
Paridad PE (par) PO (impar) 


e cambia funcionalmente segün la operación 

0 bandera a cero 

1 bandera a uno 

? bandera determinada aleatoriamente por la operación 

x caso especial; véase la nota que figura en la misma página 


Los bits de las posiciones 3 y 5 son siempre aleatorios 
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ADC А, $ 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Suma con acarreo del acumulador y el operando indicado. 


AcA+s+C<c 


s puede ser г, n, (HL), (IX + d) o (IY + d) 


27-774 
r І010|0|1|--:-- 


n byte 1: CE 


— п + 


(HL) С |||) зе 


Ох Е a aaa Е y 


24 


11010|0|1| | т |о | byte 


2 d „| byte 


Ned aaa polo] “Әу 


МЕНЕ ДЕ. 
9 9 -| byte 
r puede ser 
А — 111 
B — 000 
C — 001 
D — 010 


3: 


1: 


2 


3: 


1: 


byte 2: dato 
inmediato 


DD 

8E 

valor del desplazamiento 
FD 

8Е 


valor del desplazamiento 


E — 011 
Н- 100 
L — 101 


El operando s y la bandera de acarreo C del registro de estado 
se suman al acumulador, y el resultado se almacena en éste; s 
se define en la descripción de instrucciones ADD similares. 


ЖЖ 
қанын 
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Tiempo: 


Direccionamiento: 


Códigos byte: 
Banderas: 


Ejemplo: 


CE 


ре 


CODIGO 
OBJETO 
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5 Ciclos М Estados Т 


r 1 4 
n 2 7 
(HD) 2 7 
(IX + d) 5 19 
(ТҮ + d) 5 19 


r: implícito; n: inmediato; (HL): indirecto; (IX + d), (IY + d): 
indexado. 


ADC Ar с A B 


C D E H tL 
ae aaa 


5 H PO м с 


ее е е: е 


ADC А, 1А 
Antes: Después: 
иго тр [27222122 
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ADC НІ, ss 


Función: 


Formato: 


Descripción : 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Códigos byte: 


Banderas: 


Suma con acarreo de HL y el par de registros ss, 


HL < HL + 55 + С 


СЕРР] мет: во 
ШЕ s s] ШЕ byte 2 


El contenido del par de registros HL se suma al del par especi- 
ficado y a continuación se suma el contenido de la bandera de 
acarreo. El resultado final se almacena en HL. El par ss puede 
ser: 


BC — 00 HL — 10 
DE — 01 SP — 11 


4 ciclos M; 15 estados T; 75 useg @ 2 MHz. 
Implicito. 
$5: BC DE HL SP 


vo [e [е] 


«е |:| ее 


H se pone a 1 si hay arrastre del bit 11. 
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Ejemplo: 
БЕСТЕН 


CODIGO 
OBJETO 


18 


ADC HL, DE 


Antes: 


Después: 


СА 


Е ИИ 
"И 


Е 
L 


ADD A, (HL) 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


Suma el acumulador con la posición de memoria (HL) direccio- 
nada indirectamente. 


А + A + (HL) 


ao ЕН ИЕ... 


El contenido del acumulador se suma al de la posición de 
memoria direccionada por el par de registros HL. El resultado 
se almacena en el acumulador. 


A 


Ne 


MEMORIA 


ZI © о 


2 ciclos M; 7 estados T; 3.5 useg @ 2 MHz 


Indirecto. 


ADD A,(HL) 


Antes: Después: 


А 2 
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ADD A, (IX +d) n el acumulador con la posición de memoria indexada 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 
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(IX + d). 


A — A + (IX + d) 


ООО byte 1: DD 
[rfololo[ofr[r[e] byte 2: 86 


AA, d ——— "| byte 3: valor del desplazamiento 


El contenido del acumulador se suma al de la posición de 
memoria direccionada por el contenido del registro IX más el 
valor de desplazamiento inmediato. El resultado se almacena en 
el acumulador. 


5 ciclos M; 19 estados T: 9.5 useg @ 2 MHz. 


Indexado. 


сег еее 


Ejemplo: ADD А, (IX + 3) 


Antes: Después: 


00 0B61 04 0B61 04 

86 0B62 0B62 
овез овоз 
иа 0864 0864 


CODIGO 
OBJETO 
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ADD A, (ТУ +d) Suma el acumulador con la posición de memoria indexada 
(IY + d). 


Función: A+ A + (IY + d) 


ООО] byte 1: FD 
[ео [о [о [11] 2] byte 2: 86 
byte 3: valor del desplazamiento 


Descripción: El contenido del acumulador se suma al contenido de la posi- 
ción de memoria direccionada por el contenido del registro IY 
más el valor de desplazamiento dado. El resultado se almacena 
en el acumulador. 


Formato: 


Flujo de datos: 


Tiempo: 5 ciclos М; 19 estados T; 9.5 useg @ 2 MHz 
Direccionamiento: Indexado. 
Banderas: 


5 H РАФМ C 


911 е je|»]e 


192 


Ejemplo: ADD A, (ТУ + 1) 


Ántes: _ Después: 
АГ * | А св 
ІХ 0028 IX 002B 
ooe| 95 | 0028 | 99 | 
| 86 | 002C 002C 
[or | раа [re] 
GENS 
CODIGO 
OBJETO 
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ADD A,n 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas 


Ejemplo: 


OBJETO 
CODIGO 
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Suma el acumulador con el dato inmediato n. 


А<-А-п 


Сее byte 1: св 
[AA A ———] byte 2: dato inmediato 


El contenido del acumulador se suma al de la posición de 
memoria inmediata al código de operación. El resultado se 
almacena en el acumulador. 


А Ша 
БАБЫН 


ADD 


MEMORIA 
2 ciclos M; 7 estados T: 3.5 useg @ 2 MHz. 


Inmediato. 


S РМ м C 


eje| e| [ө[ [ө] 


ADD A, E2 


Antes: Después: 


а | ^ ЕА 


ADD А, г 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Códigos byte: 


Banderas: 
Ejemplo: 


CODIGO 
OBJETO 


Suma el acumulador con el registro r. 


А А +г 


El contenido del acumulador se suma al del registro especifica- 
do. El resultado se coloca en el acumulador; r puede ser: 


А — 111 E — 011 
B — 000 H — 100 
C — 001 L — 101 
D — 010 


l ciclo M; 4 estados T; 2 useg @ 2 MHz. 
Implícito. 


S 2 PO м 
0 


H с 
[ele g g g 
АВ Ср Е н 1 
[en [2 [вз [ва [es | 


ADD A,B 

Antes: Después 

АЙЯ 
e[ | 
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ADD HL, 55 Suma HL y el par de registros ss. 


Función: HL — HL + ss 
Formato: DDODOUUODE 
Descripción: El contenido del par especificado se suma al del par HL, y el 
resultado se almacena en HL; ss puede ser: 
BC — 00 HL — 10 
DE — 01 SP — 11 


Flujo de datos: 


Tiempo: 3 ciclos M; 11 estados T: 5.5 useg MHz. 
Direccionamiento: Implicito. 

Códigos bytes —— 86: 

Banderas: $ P/V N 


2 H с 

[|TT] | lle 

El acarreo del bit 15 pone C a 1; en caso contrario, vale 0. 
El acarreo del bit 11 pone H a 1. 


Ejemplo: ADD HL, HL 
Antes: Después: 
== H 06B1 l н ООЖ Д- 
CODIGO 
OBJETO 
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ADD IX, rr 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Códigos byte: 


Banderas: 


IT: 


Suma IX con el par de registros rr. 


X — IX 4 rr 


ШЕ ШП ЕП | byte 1: DD 
al оо 2 


El contenido del registro IX se suma al del par de registros 
especificado, y el resultado se almacena en IX; rr puede ser: 


BC — 00 IX — 10 
DE — 01 SP — 11 


ЕЕ 


x ZI 
4 ciclos M; 15 estados T: 7.5 useg @ 2 MHz. 


Implicito. 


BC DE IX SP 


рум С 
“ТТЕ 


Н se pone а 1 si hay acarreo desde el bit 11. 
С se pone a 1 я hay acarreo desde el bit 15. 
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Ejemplo: 


CODIGO 
OBJETO 
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IX 


SP 


Después: 


ADD ІҮ, гг Suma IY y el par de registros rr. 


Función: IY — IY + rr 


Formato: 


араа Бу 1: кр 
о|о]‹ + [5 о т | byte 2 


Descripción: El contenido del registro ТУ se suma al del par especificado, y el 
resultado se almacena en IY; rr puede ser: 


ВС — 00 ТУ — 10 
DE — 01 SP — 11 


Flujo de datos: 


Tiempo: 4 ciclos M; 15 estados T: 7.5 useg 2 MHz. 
Direccionamiento: Implicito. 
Códigos byte: rr: 


BC DE IY SP 


то- [09] EE 


Banderas: 
5 H PH М C 


IIT TS 


H vale 1 si hay acarreo del bit 11. 
C vale 1 si hay acarreo del bit 15. 
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Ejemplo: ADD IY, DE 


Antes: Después: 
sr ш 


CODIGO 
OBJETO 
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AND s “Y” lógica del acumulador y el operando s. 


Función: AAAS 
Formato: s puede ser: r, n, (HL), (IX + d) o (IY + d) 


" byte 1: E6 
byte 2: dato inmediato 
(HL) A6 
(X + d) byte 1: DD 
byte 2: A6 
[————4————] byte 3: valor del desplazamiento 
(IY + d) byte 1: FD 


byte 2: A6 


A byte 3: valor del desplazamiento 


г puede ser 
А — 111 E — 011 
B — 000 H — 100 
C — 001 L — 101 
D — 010 
Descripción: El acumulador y el operando especificado se someten a la 


operación lógica “Y” (AND), y el resultado se almacena en el 
acumulador; s se define en la descripción de instrucciones ADD 
similares. 


Flujo de datos: 
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Tiempo: 


Direccionamiento: r: implícito; n: inmediato; (HL): indirecto; (IX + d), (IY + d): 
indexado. 
Códigos byte: AND r «A в 


C D E H L 
Гоа 


Banderas: s z н Фмн с 
[ele] |:| е оо 
Ejemplo: AND 4B 


Antes: Después: 


АГ » | А Ж 


CODIGO 
OBJETO 
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BIT b, (HL) Verifica el bit b de la posición de memoria (HL) direccionada 
indirectamente. 


Función: Z < (HL), 


Formato: 


CCE bve 1: св 
CETTE bye 2 
Descripción: Se verifica el bit especificado de la posición de memoria direc- 


cionada por el contenido del registro HL y se activa la bandera 
Z en función del resultado; b puede ser: 


0 — 000 4 — 100 
1 — 001 5 — 101 
2 — 010 6 — 110 
3 — 011 7 — 111 


Flujo de datos: 


| Tiempo: 3 ciclos М; 12 estados T: 6 useg @ 2 MHz. 
Direccionamiento: Indirecto. 


Banderas: 
5 PNN с 


2 H 
['[ө| ['] [eie] | 


Códigos byte: 


b 0 1 2 3 4 5 6 7 
св. [46] a| so | se | 66 ЕЗ 


` Ejemplo: 


CODIGO 
OBJETO 


BIT 3,(HL) 


Antes: 
Е 
н L 
6442 ЕСЕН 
PONENS 


Después: 


7 t F 


BIT b, (IX + d) Verifica el bit b de la posición de memoria indexada (IX + d). 


Función: Z — (IX + d), 


Formato: BERCRERERE [o [1] byte 1: DD 
[| byte 2: СВ 
[A а------ byte 3: valor del desplazamiento 


reae. [i E byte 4: 
Descripción: El bit especificado de la posición de memoria direccionada por 


el contenido del registro IX más el valor del desplazamiento 
dado se verifica, y se activa la bandera Z en función del resulta- 
do; b puede ser: 


0 — 000 4 — 100 
1 — 001 5 — 101 
2 — 010 6 — 110 
3 — 011 7 — 111 


Flujo de datos: 


ALU 


тоор 
е мот 


Тіетро: 5 ciclos М; 20 estados Т: 10 useg @ 2 MHz. 


Direccionamiento: Indexado. 


Códigos byte: b o 1 2 3 4 5 6 7 
cocos: [Теа [ры | 
Banderas: s z H Рум C 
elei [| Ifo] | 
Ejemplo: BIT 6, (IX + 0) 
Antes: Después: 
F IÓN 
ІХ ААП IX ААП 
[ 55 | ми БЕГЕН si 
| ce | [~] | 
[о | 
E) 
CODIGO 
OBJETO 


BIT b, (ТҮ + d) Verifica el bit b de la posición de memoria indexada (ТУ + 9). 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


[PPLPPLDIeD] byte 1: FD 
[Оо] bye 2: СВ 

byte 3: valor del desplazamiento 
bute 4 


Se verifica el bit especificado de la posición de memoria direc- 
cionada por el contenido del registro ГУ más el valor del despla- 
zamiento dado y se activa, en consecuencia, la bandera Z; b 
puede ser: 


O— 000  4-100 
1-001 5-10 
2—00 6-10 
3-01 7-11 


I UOU V > 


5 ciclos M; 20 estados T: 10 useg @ 2 MHz. 
Indexado. 
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Códigos byte: 
Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


0 12 3 4 5 6 7 
поза (46| ee [se] [46 в [re [7e] 


5 2 H РУ N C 
е ТР | 
ВІТ 0, (IY + 1) 

Antes: 


[ 3*2 | 
М 12 | 


Después: 


mi to | 
ніз| 8 | 


ВИТ b,r 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Códigos byte: 


Verifica el bit b del registro г. 


Ze га 
о [о [efi Г | byte 1: CB 
о |а peor byte 2 


Verifica el bit especificado del registro dado y activa la bandera 
cero en función del resultado; b y r pueden ser: 


b: 0 — 000 4 — 100 
1 — 001 5 — 101 
2 — 010 6 — 110 
3—011 7 — 111 

r: А — 111 E - 011 
B — 000 H — 100 
C — 001 L — 101 
D — 010 


2 ciclos M; 8 estados T: 4 useg @ 2 MHz. 


Implicito. 
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Banderas: 


Ejemplo: 


| s | 
y] 


CODIGO 
OBJETO 
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5 2 H РМ N С 
RARES 


BIT 4, В 
Antes: Después: 


a] PU | ва | БИ; 


CALL сс, pq 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Llamada condicional a subrutina. 


Si cc cierto: (SP — 1) < PC,,; (SP — 2) < РС; SP < SP -2; 
PC < pq 
Si сс falso: PC + PC + 3 


а ГоТо | byte 1 
A byte 2: dirección, byte inferior 


4-1] byte 3: dirección, byte superior 


Si se satisface la condición, el contenido del contador del pro- 
grama se empuja en la pila tal como se describe en las instruc- 
ciones PUSH. А continuación, el contenido de la posición de 
memoria inmediatamente siguiente al código de operación se 
carga en la parte inferior del PC y el contenido de la posición 
de memoria siguiente se carga en la mitad superior del PC. La 
siguiente instrucción se tomará de esta nueva dirección. Si la 
condición no se satisface, se ignora la dirección pq y se ejecuta 
la instrucción siguiente; cc puede ser: 


NZ — 000 PO — 100 
Z — 001 PE — 101 


NC — 010 P — 110 
C — 011 M — 111 


Al final de la subrutina llamada puede usarse una instrucción 
RET para restablecer el PC. 


SP ,À 7, 
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Tiempo: 


| useg 

Ciclos M Estados T @2 MHz 
condición 

cierta 5 17 8.5 
condición 
falsa 3 10 5 
Direccionamiento: Inmediato 
Códigos byte: CC: NZ.Z NC C PO PE Р M 


Серые Te rere] v 


Banderas: 5 2 н ммм С 
(efecto nulo) 
Ejemplo: 7 CALL 7, В042 
Antes Después 
9), І 
PC ес ИИА 
se s 
8810 во _ | 
Е в ви 64 | wnf os | 
| ño | СІР 32 | 8812 
CODIGO 
OBJETO 
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CALL ра Llamada а subrutina a la posición pq. 


Función: (SP — 1) PC 


sup? 


(SP — 2) < РС; SP < SP — 2; PC < pq 


Formato: 


[9] byte 2: dirección, byte inferior 


1 
AM NA byte 3: dirección, byte superior 


1 IA byte 1: CD 


Descripción: El contenido del contador del programa se empuja en la pila tal 
como se describe en las instrucciones PUSH. A continuación se 
carga el contenido de la posición de memoria siguiente al códi- 
go de operación en la mitad inferior del PC y el de la posición 
siguiente en la mitad superior del PC. La instrucción siguiente 

“se traerá de esta nueva dirección. 


Flujo de datos: 


— 


==) 


Tiempo: 5 ciclos M; 17 estados T; 8.5 useg @ 2 MHz. 
Direccionamiento: Inmediato. 
Banderas: 


5 Z H рум с 


ССТТ («сю шо) 


213. 


Ejemplo: CALL 40B1 


Antes: Después: 
еГ año |) АГА 
8121 — 9A ов! 
E ж а а А 
[~] [~] 


CODIGO 
OBJETO 
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CCF Complementación de la bandera de acarreo. 


Función: C ec 

F to: 

OUT ponnani- 
Descripción: Se complementa la bandera de acarreo. 


Flujo de datos: 


Tiempo: 1 ciclo M; 4 estados T; 2 useg @ 2 MHz. 


Direccionamiento: Implicito. 


Banderas: Z H PNN с 
| [I | lole 
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CPs Comparación del operando s con el acumulador. 


Función: А – 5 
Formato: s puede ser г, n, (HL), (IX + d) o (IY + d). 
Ex Ee xs eee 
а [+Pof+]+]:[+[+[0] FE 
byte 2: dato inmediato 
(HL) |1|913|1|3|11110| byte 1: BE 
ax d перо] byte 1: DD 
LP [ol+[+]+[>[+[0] вие 2: BE 
byte 3: valor del desplazamiento 
ava PEPPI iiel] byte 1: FD 
ШОО] byte 2: BE 
byte 3: valor del desplazamiento 
r puede ser: 
А — 111 E — 011 
B — 000 H — 100 
C — 001 L — 101 
D — 010 
Descripción: El operando especificado se resta del acumulador, y el resultado 
se descarta: s se define en la descripción de instrucciones ADD 


similares. 
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Flujo de datos: 


Tiempo: 
5 Ciclos М Estados T o? MHz 
r 1 4 2 
n 2 7 3.5 
(HL) 2 7 3.5 
(IX + d) 5 19 9.5 
(IY + d) 5 19 9.5 | 
Direccionamiento: г: implicito; n: inmediato; (HL): indirecto; (IX + d), (IY + d): 
indexado. 
Códigos byte: CP r: с A B C D E НЕ 
CICICICYCICICI 
ie ГІСІЕСЛЕС [ә 
Ejemplo: CP (HL) 
Antes: Después: 
e БЕСТЕН) АГ» Ж 
ве (газет) 
ЕСІ ва | в БЕКЕН 
E) A] и 
CODIGO 
OBJETO 
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CPD 


Función: 


Formato: 


Descripcción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 


¿lic 
FE 
| 


CODIGO 
OBJETO 
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Comparación con decremento. 


A — [HL]; HL < HL - 1; ВС < BC- 1 
ао ао) byte 1: ED 
[o] +[o[+[ofo[»] byte 2: A9 


El contenido de la posición de memoria direccionada por el par 
de registros HL se resta del contenido del acumulador y se 
descarta el resultado. À continuación se decrementan los pares 
de registros HL y BC. 


A d 
8 с 
D Е 


— —Q 


4 ciclos M; 16 estados Т; 8 useg @ 2 MHz. 
Indirecto. 
Poner a 0 si BC = 0 tras la ejecución; 


5 2 
|ө[х| Jej |х] | dejar a 1 en caso contrario. 
Hacer 1 si A = (HL) 


CPD 
Antes: Después: 
А [| œ% | JE ИЕ 
c CHE 
н t ГРІ 
A Е 
sens 
p | 


CPDR 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Banderas: 


Comparación de bloques con decremento. 


A — [HL]; HL + HL — 1; ВС e ВС — 1; 
Repetir hasta que ВС = 0 o A = (HL). 


Пот} byte 1: ED 
| [of+[+[+[ofo[+] byte 2: во 


El contenido de la posición de memoria direccionada por el par 
de registros HL se resta del contenido del acumulador y el 
resultado se descarta. А continuación se decrementan los pares 
de registros BC у HL. Si BC + 0 y A + (НІ), el contador del 
programa se decrementa en dos, y la instrucción vuelve a ejecu- 
tarse. 


H 


L 


BC = 0 o A = (HL): 4 ciclos M; 16 estados T; 8 useg @ 2 
MHz. 
BC £0 y A + (HL): 5 ciclos M; 21 estados T; 10.5 useg (0 
2M Hz. 


Poner a 0 si BC = Otras la ejecución; 
deja a 1 en caso contrario. 


Dejar a 1 si A — (HL). 
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Ejemplo: 


CODIGO 
OBJETO 
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Después: 


H 


CPI 
Función: 
Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


Comparación con incremento. 
A — [HL]; HL HL + 1; ВС < BC — 1 


E тт о] [т] о 1] byte 1: ED 
ШЕШЕ 0 ol] byte 2: Al 
El contenido de la posición de memoria direccionada por el par 


de registros HL se resta del contenido del acumulador y se 
descarta el resultado. El par de registros HL se incrementa, y el 


BC se decrementa. 
ГГ 


— y 


А 


C NER ПВС an == | 
нЙ Ы 1 = 
4 ciclos М; 16 estados Т; 8 useg @ 2 MHz. 
Indirecto. 
| Poner а 0 я ВС =0 tras la eje- 
РМ М cución; dejar a l en caso contrario. 
ФРГ ФГ [ГГ Dejar a 1 si A = (HL). 
CPI 
Antes: Después: 
Ly аа 
B c B * ИЕ A с 
HE ГР 
¡AR 
8689 | 9B | 8689 
-J е 
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CPIR 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 
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Comparación de bloques con incremento. 


A — [HL]; HL < HL + 1; ВС < ВС - 1; 
Repetir hasta que BC — 0 o A = (HL). 


HBHBDOBHBDD byte 1: ED 
Dile [lefele] byte 2: B1 


El contenido de la posición de memoria direccionada por el par 
de registros HL se resta del contenido del acumulador y el 
resultado se descarta. A continuación se incrementa el par de 
registros HL y se decrementa el ВС. Si ВС + 0 y A + (HL), el 
contador del programa se decrementa en dos y la instrucción 
vuelve a ejecutarse. 


F 


BC=0 0 A =(HL): 4 ciclos М; 16 estados T: 8 useg @ 2 
MHz. 
BC + 0 y A + (HL): 5 ciclos M; 21 estados T: 10.5 useg @ 2 
MHz. 


Indirecto. 


Poner a 0 я BC = 0 tras la eje- 
cución; fijar а 1 en caso contrario. 


Fijar a 1 si A = (HL) . 


Ejemplo: 


CODIGO 
OBJETO 


Después: 


B 


ос в | 
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CPL. Complementar el acumulador. 


Funcién: À +- À 

Formato: реГ 2Е 

Descripción: El contenido del acumulador se complementa (se invierte) y el 
resultado vuelve a almacenarse en el acumulador (complemen- 
to a 1). 


Flujo de datos: 


A 
B с 
D E 
H ( 
Tiempo: 1 ciclo M; 4 estados T; 2 useg @ 2 MHz. 
Direccionamiento: Implicito. 
Banderas: s z H PVN с 
| 1 | 1 | 
Ејетріо: CPL 
Ántes: Después: 
> 
CODIGO 
OBJETO 
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DAA 
Función: 


Formato: 


Descripción: 


Flujo de datos: 


Ajuste decimal del acumulador. 


Véase a continuación. 


о10 | ъ | 1 | 1 27 


La instrucción suma condicionalmente “6” al nibble derecho, o 
al izquierdo, o a los dos, del acumulador, según el registro de 
estado, para realizar la conversión BCD tras las operaciones 


aritméticas. 
Número 
N C Valor del Valor del suniado-d с tras la 
nibble sup. nibble inf. A ejecución 
0 0 0-9 0 0-9 00 0 
(ADD, | 0 0-8 0 A-F 06 0 
ADC, | 0 0-9 1 0-3 06 0 
INC) | 0 A-F 0 0-9 60 1 
0 9-Е 0 А-Е 66 1 
0 А-Е 1 0-3 66 1 
1 0-2 0 0-9 60 1 
1 0-2 0 A-F 66 1 
1 0-3 1 0-3 66 1 
1 0 0-9 0 0-9 00 0 
(SUB, | 0 0-8 1 6-F FA 0 
SBC, | 1 7-F 0 0-9 AO 1 
DEC, | 1 6-F 1 6-F 9A 1 
NEG) 


А 


LA | 


F 
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Tiempo: 1 ciclo M; 4 estados T; 2 useg @ 2 MHz. 


Direccionamiento: Implícito. 
Banderas: 2 H Фунс 
өө! е le, je] 
Ejemplo: DAA 
Antes: Después: 
A[ m Г ж jr 2722 2: 
E 1 
CODIGO 
OBJETO 
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DEC m 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


(IY + d) 


Decrementa el operando m. 


пет- 1 


m puede ser г, (HL), (IX + 9), (IY + d). 


ЖИЫ ЕТЕШ AE 
ее ЮВЕ 
EXER К 
кы ше е 
ЕЕ ЕРЕШЕ 
ЕН БЕР 
ІШЕ СЕЛ Ей 


г puede ser: 


А — 111 
B — 000 
C — 001 
D — 010 


: DD 


539 


: FD 


: 35 


E — 011 
H — 100 
L — 101 


: valor del desplazamiento 


: valor del desplazamiento 


El contenido de la posición direccionada por el operando espe- 
cificado se decrementa y vuelve a almacenarse en esa posición; 
m se define en la descripción de instrucciones INC similares. 


r oo» 


- m 


227 


Tiempo: 


Direccionamiento: 
Códigos byte: 
г: 
Banderas: 
Ejemplo: 


CODIGO 
OBJETO 
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: useg 
m Ciclos M Estados T @ 2 MHz 
r 1 4 2 
(HL) 3 11 5.5 
(IX + d) 6. 23 11.5 
(IY + d) 6 23 11.5 


r: implícito; (HL): indirecto; (IX + d), (IY + d): indexado. 


DEC r 

A B C D E H 1 

[80 [os [oo 15 [1] 25] 2] 

S 2 H PAN C 
ее! jej е | | 

DEC С 

Antes: Después: 
ИСИ: EZ 


DEC rr 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Códigos byte: rr: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


Decrementa el par de registros rr. 


rerr—1 


Se decrementa el contenido del par de registros especificado, y 
el resultado vuelve a almacenarse en ese mismo par; rr puede 
ser: 


BC — 00 HL- 10 
DE — 01 SP — 11 


1 ciclo M; 6 estados T; 3 useg @ 2 MHz. 
Implícito. 


BC DE HL SP 


Los [тв as || 


S Z H PN N C 
(efecto nulo) 
DEC BC 
Antes: Después: 
A E 2229872226 
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DEC IX Decrementa IX. 


Función: IX —IX— 1 


ШЕ [o] EE Г | byte 1: DD 
RETARDO вие 2: 28 


Descripción: Se decrementa el contenido del registro IX, y el resultado vuelve 
a almacenarse en IX. 


Formato: 


Flujo de datos: 


Tiempo: 2 ciclos М; 10 estados T; 5 useg @ 2 MHz. 
Direccionamiento: Implicito. 
Banderas: 
s z H РУМ с 
1 T Я 1] (efecto nulo) 
Ejemplo: DEC IX 
Antes: Después: 
~ хан к 
ШИШ 
ШЕЛ 
> 
CODIGO 
OBJETO 
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DEC IY 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 
Ejemplo: 


| ro | 
2 | 
| 


CODIGO 
OBJETO 


Decrementa IY. 


IY -IY — 1 


ap apa | ! ОШ byte 1: FD 
Са} вие 2: в 


Se decrementa el contenido del registro IY, y el resultado vuelve 
a almacenarse en IY. 


тоо > 


2 ciclos M; 10 estados T; 5 useg @ 2 MHz. 


Implicito. 
5 27 H РУ М С 
| | | (efecto nulo) 
DEC IY 
Antes: Después: 
w[ ] v SET 
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DI 


Función: 


Formato: 


Descripción: 


Tiempo: 


Direccionamiento: 


Banderas: 
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Invalida interrupciones. 


IFF —0 


БЕ ЕШШ 


El biestable de interrupciones se pone a 0 y, en consecuencia, se 
impiden todas las interrupciones enmascarables. Vuelve a vali- 
darse con una instrucción El. 

1 ciclo М; 4 estados T; 2 useg @ 2 MHz. 


Implicito. 


5:22 H PN NC 


ГІТІТІТІІ eue по 


DJNZ e Decremento de B y, si no es cero, salto relativo de longitud e. 


Función: Ве В-1; я B£0: PC-PC+e 


Formato: olololi о [о [о o | byte 1: 10 
i 2—4 byte 2: valor del desplazamiento 


Descripción: Se decrementa el registro B. Si el resultado no es cero, se suma 
el valor del desplazamiento al contador del programa en aritmé- 
tica de complemento a dos, para permitir saltos hacia adelante y 
hacia atrás. El valor del desplazamiento se suma al PC + 2 (tras 
el salto); en consecuencia, el desplazamiento efectivo va de — 126 
a +129 bytes. El ensamblador resta automáticamente del valor 
del desplazamiento fuente para generar el código hexadecimal. 


Flujo de datos: 


LOGICA 
DE CONTROL 


Tiempo: B + 0: 3 ciclos М; 13 estados T; 6.5 useg @ 2 MHz. 
B=0: 2 ciclos М; 8 estados T; 4 useg @ 2 MHz. 
Direccionamiento: Inmediato. 
Banderas: 5 2 H РУУ М е 
m | Е LI | | | (efecto nulo) 
Ejemplo: ОЛМА $ — 5 ($ = PC actual) 
Antes: Después: 
‚| 
CODIGO 
OBJETO 
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El 


Función: 


Formato: 


Descripción: 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 
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Habilita interrupciones. 


IFF + 1 


И ЕЮГЕ авв 


El biestable de interrupciones se pone а 1 y, en consecuencia, 
permite que se produzcan interrupciones enmascarables tras la 
ejecución de la instrucción que sigue a la El. En ese tiem- 
po se invalidan las interrupciones filtrables. 


1 ciclo M; 4 estados T; 2 useg @ 2 MHz. 
Implícito. 


S Z H PH М C 


ГІТІТТІТІГ ене mu) 


Una secuencia habitual al término de una rutina de interrup- 
ción sería: 

El 

RETI 

La interrupción enmascarable vuelve a habilitarse al término de 
RETI. 


EX AF, АР" 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


Intercambia acumulador y banderas con los registros alternati- 
vos. 


АЕ © AF 


ЕНЕ; 


Los contenidos del acumulador y del registro de estado se 
intercambian con los del acumulador y el registro de estado 
alternativos. 


1 ciclo M; 4 estados T; 2 useg @ 2 MHz. 
Implicito 


5 2 H рум C 


00000000 


EX AF, АЕ’ 

Antes: Después: 

АС в a jF А РЗ, 
A[ 9 за ]r А ЙҮ ҮЙҮҮ к 
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EX DE, HL 


Función: 


Formato: 
Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


29800 


Intercambia los registros HL y DE. 


DE — HL 


тъ {то тт оа ЕВ 


Se intercambian los contenidos de los dos pares de registros DE 
y HL. 


1 асю M; 4 estados T; 2 seg ( 2 MHz. 


Implicito. 

5 2 H PN м с 
CTILII ео nuo 
ЕХ DE, HL 

Antes: Después: 


AME6 E D —— ЖАЛ е 
9604 L 222/222: 


EX (SP), HL Intercambia HL con el .elemento superior de la pila. 


Función: (SP) o L; (SP + 1) < Н 
Formato; 000000009 
Descripción: El contenido del registro L se intercambia con el de la posición 


de memoria direccionada por el puntero de la pila. El conte- 
nido del registro H se intercambia con el de la posición de 
memoria inmediatamente siguiente a la direccionada por el 
puntero de la pila. 


Flujo de datos: 


Tiempo: 5 ciclos M; 19 estados T; 9.5 useg @ 2 MHz. 
Direccionamiento: Indirecto. 
Banderas: S 72 H PVN C 
L] | | | | | | | (efecto nulo) 
Ejemplo: EX (SP), HL 
Después: 
Antes: 
H 8290 L ЖҰТ! 
» 549 > ЕЕЕ”: — ] 
Шығын (ел; 
7 s | 8400 ж? 
Y) ж БЕКЕН B40A WEY 
OBJETO Y) А 
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EX (SP), IX 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 
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Intercambia IX con el elemento superior de la pila. 
(SP) o IX;,,; (SP + 1) o IX, 


ПП [о] | byte 1: DD 
[+] +[ofofof+[+] byte 2: ЕЗ 


El contenido de la mitad inferior del registro IX se intercambia 
con el de la posición de memoria direccionada por el apuntador 
de la pila. El de la mitad superior de IX se intercambia, a su 
vez, con el de la posición de memoria inmediatamente siguiente 
a la direccionada por el apuntador de la pila. 


ССС 


6 ciclos M; 23 estados T; 11.5 useg @ 2 MHz. 
Indirecto. 


5 Z Н рум C 


| | (efecto nulo) 


Ejemplo: EX (SP), IX 


Antes: Después: 

хм] қ Яо ар 7, 

99 м] 
| DD | 0402] 68 | 00577 
003 or | ов 77777 


CODIGO 
OBJETO 


EX (SP), IY 
Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 
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Intercambia IY con la parte superior de la pila. 


(SP) o 1Y,,,; (SP + 1) «ТҮ, 


ор] byte 1: FD 
+] a] rfofolo[r]*] byte 2: ЕЗ 


El contenido de la mitad inferior del registro ТУ se intercambia 
con el de la posición de memoria direccionada por el apuntador 
de la pila. El de la mitad superior de IY se intercambia, a su 
vez, con el de la posición de memoria inmediatamente siguiente 
a la direccionada por el apuntador de la pila. 


6 ciclos M; 23 estados T; 11.5 useg @ 2 MHz. 
Indirecto. 


s z H мм N C 
(efecto nulo) 


Ejemplo: ЕХ (SP), IY 
Antes: Después: 
"ZZ 
Г | 


ла AN T. 
Г & | 6212 4D 6212 AA 
AW au 


CODIGO 
OBJETO 
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EXX 


Función: 


Formato: 
Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 
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Intercambia registros alternativos. 


ВС o ВС’; DES DE; HL + НИ 


Ara aaa bs 


Los contenidos de los registros de tipo general se intercambian 
con los correspondientes registros alternativos. 


1 ciclo M; 4 estados T; 2 useg @ 2 MHz. 


Implicito. 

5-2 Н PV М с 
EEE (efecto nulo) 
EXX 

Antes: Después: 


HALT 


Función: 


Formato: 


Descripción: 


Tiempo: 


Direccionamiento: 


Banderas: 


Detiene la CPU. 


La CPU deja de actuar. 


БЕГ] 76 


La CPU deja de funcionar y ejecuta instrucciones МОР conti- 
nuamente, para proseguir con los ciclos de refresco de la memo- 
ria, hasta que recibe una interrupción o una orden de reinicio. 


1 ciclo M; 4 estados T; 2 useg @ 2 MHz + el tiempo necesario 
para ejecutar un nümero indefinido de instrucciones NOP. 


Implicito. 


5 2 H PN N C 


ГІТІТІТТ! efecto nuto) 
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IM O 


Función: 


Formato: 


Descripción: 


Tiempo: 


Direccionamiento: 


Banderas: 


244 


Activa el modo de interrupción 0. 


Control interno de interrupciones, 


11а јо [т [јоз | byte 1: ED 


CHELLES bre 2: 46 


Activa el modo de interrupciôn 0. En esta situaciôn, el dispositi- 
vo interruptor puede dejar una instrucción en el bus de datos 
para su ejecución; el primer byte de dicha instrucción debe 
aparecer durante el ciclo de identificación de la interrupción. 


2 ciclos M; 8 estados T; 4 useg @ 2 MHz. 


Implícito. 


IM 1 Activa el modo de interrupción 1. 


Función: Control interno de interrupciones. 


Formato: , E Л ШЕ То | byte 1: ED 
Lelie ieli [110] byte 2: 56 


Descripción: Activa el modo de interrupción 1. Cuando se produce la inte- 
rrupción, se ejecuta una instrucción RST 0038H. 


E 
0038 


(en el momento de 
la interrupción) 
е 


Flujo de datos: 


--- 
PILA 
Tiempo: 2 ciclos M; 8 estados T; 4 useg а 2 MHz. 
Direccionamiento: Implicito. 
Banderas: s 2 H PAN с 


СІТІТІТІІ ee mio 
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IM 2 Activa el modo de interrupción 2. 


Función: Control interno de interrupciones. 


Formato: Пт] ®] "| *| ®] "| byte 1: ED 
0 ae ale byte 2: 5Е 


Descripción: Activa el modo de interrupción 2. Cuando se produce la inte- 
rrupción, el periférico utilizado debe entregar un byte como 
parte inferior de una dirección; la parte superior del vector de 
dirección procede del contenido del registro I. Este señala una 
segunda dirección almacenada en memoria, que se carga en el 
contador del programa, tras lo cual comienza la ejecución. 


Tiempo: 2 ciclos М; 8 estados T; 4 useg (9 2 MHz. 
Direccionamiento: Implícito. 
Banderas: 5 7 н_ рум C 


[ТЇГЇ Г 1 feto nuo 
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IN r, (C) 
Función: 
Formato: : 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Códigos byte: 


Banderas: 


Carga el registro r a partir del puerto (C). 
r — (C) 


AA os 
Го ес оо [о] Буве 2 


Se lee el dispositivo periférico direccionado рог el contenido del 
registro C, y el resultado se carga en el registro especificado. 
C proporciona los bits AO a A7 del bus de direcciones. 

В proporciona los bits A8 a А15. 


^ PUERTO 
B с 
D E 
H L 
r puede ser: 
A — 111 E — 011 
B — 000 H — 100 
C — 001 L — 101 


D — 010 


3 ciclos M; 12 estados T; 6 useg @ 2 MHz. 


Externo. 


r A B C D E H L 
peras aaa 


5 z H Фумс 
ejej [е Өө: 
Es importante señalar que IN A,(N) no ejerce ningún efecto 


sobre las banderas, al contrario que IN r,(C), que sí lo ejerce. 
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Ejemplo: 


CODIGO 
OBJETO 
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IN D,(C) 
Antes: 
c 


A5 


Después: 


Са k 
оЙйғА PUERTO 


IN A, (N) 
Función: 
Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 


2 


CODIGO 
OBJETO 


Carga el acumulador a partir del puerto de entrada N. 


E 1 JEN [o] Г] byte 1: DB 
= -] byte 2: dirección puerto 


Se lee el dispositivo periférico N, y el resultado se carga en el 
acumulador. 

El literal М se sitúa en las líneas AO a A7 del bus de direcciones. 
А proporciona los bits А8 a А15. 


c 
: С] 
b PUERTO ER 


3 ciclos M; 11 estados T; 5.5 useg @ 2 MHz. 


ош” > 


Externo. 

52 н РУМ C 

[ | zl IST | Ж (efecto nulo) 
IN А, (В2) 

Antes: Después: 


А 8$ ][ m Jruerto АДА [Juno 


B2 
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INC r 


Función: 


Formato: 


Descripción: 


Flujos de datos: 


Tiempo: 


Direccionamiento: 


Códigos byte: 


Banderas: 


' Ejemplo: 


> 
~oo 
CODIGO 


OBJETO 
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Incrementa el registro r. 


rer+l 


El contenido del registro especificado se incrementa; r puede 


ser: 

А — 111 E — 011 
B — 000 H — 100 
C — 001 L — 101 
D — 010 

A 

B c 

D E 

H L 


1 ciclo M; 4 estados T; 2 useg @ 2 MHz. 


Implicito. 


I: A B C D E 


н 1 
ре 


5 2 н PN с 
ее! || е 

INC D 

Antes: Después: 


оГ» | о Ж 


INC rr Incrementa el par de registros rr. 


Función: rerr+1 
PME [еер 
Descripción: El contenido del par de registros especificado se incrementa, y el 
resultado se almacena de nuevo en el mismo par; rr puede ser: 
BC — 00 HL — 10 
DE — 01 SP — 11 


Flujo de datos: 


Tiempo: 1 ciclo M; 6 estados T; 3 seg @ 2 MHz. 
Direccionamiento: Implícito. 
Códigos byte: IT: BC DE HL SP 
EIEIEIEI 
Banderas: S z H PN N с 
ЕТТІТІЗТТІГІ esee шо 
Ejemplo: INC HL 
Antes: Después: 
нм н РЫ РУ | 
CODIGO 
OBJETO 
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INC (HL) 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 


Ejemplo: 


34 


CODIGO 
OBJETO 
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Incrementa la posición de memoria direccionada indirectamente 
(HL). 


(HL) — (HL) + 1 


elebi ehil] 34 


Se incrementa el contenido de la posición de memoria direccio- 
nada por el par de registros HL, y el resultado se almacena de 
nuevo en dicha posición. 


3 ciclos M; 11 estados T; 5.5 useg @ 2 MHz. 
Indirecto. 


5 H PAD М 


ее е те 


INC (HL) 


INC (IX + d) 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 


Incrementa la posición de memoria indexada (IX + d). 


(IX + а) € (IX + d) + 1 


ШЕШШ ИШ E ы 
cae ИИ E. 
byte 3: valor del desplazamiento 


Se incrementa el contenido de la posición de memoria direccio- 
nada por el registro IX más el valor del desplazamiento dado, y 
el resultado vuelve a almacenarse en dicha posición. 


га 


Фр 
[DATO | 


SS 


6 ciclos M; 23 estados T; 11.5 useg @ MHz. 
Indexado. 


5 H PAN C 


еее е | 
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Ejemplo: INC (IX + 2) 


Antes: 


CODIGO 
OBJETO 
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INC (IY + а) Incrementa la posición de memoria indexada (ТҮ + d). 


Función: (IY + d) < (IY + d) + 1 


мш Ре byte 1: FD 
Lofofr[rfofrfofoj byte 2: 34 
byte 3: valor del desplazamiento 


Descripción: Se incrementa el contenido de la posición de memoria direccio- 
nada por el registro ГУ más el valor del desplazamiento dado, y 
el resultado se almacena de nuevo en la misma posición. 


Flujo de datos: 


Tiempo: 6 ciclos М; 23 estados T; 11.5 useg @ 2 MHz. 
Direccionamiento: Indexado. 
Banderas: «е вее: 
Ejemplo: INC (1Y +0) 
Antes: Después: 
w[ om |] оо) 
оо 51 | о Р 
ШЕГЕН оо в | of" во | 
| 0 | БЕЛЕК 
— jJ 
CODIGO 
OBJETO 
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INC IX Incrementa IX. 


Función: IX — IX +1 
Formato: 
CERDEBETDT] bre 1: DD 
о |о | т [о |о |о | 1 1 Буе 2: 23 
Descripción: Se incrementa el contenido del registro IX, y el resultado vuelve 


a almacenarse en IX. 


Flujo de datos: 


ооо» 


Tiempo: 2 ciclos М; 10 estados T; 5 seg @ 2 MHz. 
Direccionamiento: Implícito. 
Banderas: 

s z H PV N с 


(efecto nulo) 


Ejemplo: INC IX 


Antes: Después: 
X х ЕУ А 
3 


CODIGO 
OBJETO 
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INC lY Incrementa IY. 


Función: IY <IY +1 


a ae byte 1: FD 
[o[o[+[o[o[of+[1] byte 2: 23 


Descripción: Se incrementa el contenido del registro ГУ, y el resultado vuelve 
a almacenarse еп IY. 


Formato: 


Flujo de datos: 


оо» 


ІҮ 


Tiempo: 2 ciclos М; 10 estados T; 5 useg @ 2 MHz. 
Direccionamiento: Implicito. 
Banderas: T Н EON TE 

EET E deseo d (efecto nulo) 
Ejemplo: INC IY 

Antes: Después: 


CODIGO 
OBJETO 
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IND 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 
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IND 
Antes: Después: 
в с 2—02 85 (с 
н O6BA L н Ы 
PUERTO PUERTO 
B5 B5 


oaj 00 | 068^ 


Entrada con decremento. 


(НІ) < (С); B- B- 1; НЕ < НІ ~ 1 


тт о [ът [о тр byte 1: ED 
"|011 opo]: Te] byte 2: AA 


Se lee el dispositivo periférico direccionado por el registro C, y 
el resultado se carga en la posición de memoria direccionada 
por el par de registros HL. A continuación se decrementan el 
registro B y el par HL. 


(22271 
enze —— 


(ono 
Z 
с —— =й 


Е 
L 


4 ciclos M; 16 estados T; 8 useg 2 MHz. 


Externo. 


5 


DONE ES activa a 1 si B = 0 tras la eje- 


cución. 
Se pone a 0 en caso contrario. 


INDR 
Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


Entrada de bloque con decremento. 


(HL) = (C; В < В – 1; HL- HL ~ 1 
Repetir hasta que B=0 


СЕС Г byte 1: ED 
byte 2: BA 


Se lee el dispositivo periférico direccionado por el registro C, y 
el resultado se carga en la posición de memoria direccionada 
por el par de registros HL. A continuación se decrementan el 
registro B y el par HL. Si B no es cero, el contador del 
programa se decrementa en 2, y la instrucción vuelve a ejecu- 
tarse. 


At es | 
B[CONTADOR] de 


E PUERTO 


Бане рва 
бі 


В = 0: 4 ciclos M; 16 estados T; 8 useg @ 2 MHz. 
B + 0: 5 ciclos M; 21 estados Т; 10.5 useg @ 2 MHz. 


Externo. 
H P. 
ӘЛИДЕН 
INDR 
Antes: Después: 
e[ o» Ге (с SUA 3 |с 
(ов Y VLL 
PUERTO PUERTO 
56 56 
ск Са 
ою 8 | 
ок 48 | a 


ое ЗА | С 
А-- a 


INI 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 
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Entrada con incremento. 


(НІ) « (С); BCB— 1; Н < НІ + 1 


аа о Те | byte 1: ED 
Го [о То То То | byte 2: A2 


Se lee el dispositivo periférico direccionado por el registro С, y el 
resultado se carga en la posición de memoria direccionada por 
el par de registros HL. El registro B se decrementa y el par de 
registros HL se incrementa. 


El contenido de C se deja en la mitad inferior del bus de 
direcciones y el de B en la superior. La selección de E/S suele 
hacerse mediante C, es decir, mediante AO a A7. B es el conta- 
dor de byte. 


B 


| 
ИБН | 
A a | PUERTO 
ГЇ 


L 


Ши 
NT 
E 


H 


4 ciclos М; 16 estados T; 8 useg @ 2 MHz. 


Externo. 


5 Z H PY N € 
SONGS 


Z se activa a 1 si B = O tras la ejecución. 
Se pone a 0 en caso contrario. 


Ejemplo: 


CODIGO 
OBJETO 


Después: 


ZZ s |с 
SIA 
[ = еценто 


56 


6А 
7 


NEM 


po 
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INIR Entrada de bloque con incremento. 


«Función: (HL) < (С); Be B – 1; HL + HL + 1; repetir hasta que В 
=0 


Formato: E 1 | то | 1 JE || byte 1: ED 


1|0)1 9|0|1|0| byte 2: B2 


Descripción: Se lee el dispositivo periférico direccionado por C, y el resultado 
se carga en la posición de memoria direccionada por el par de 
registros HL. El registro B se decrementa y el par HL se 
incrementa. Si B no es cero, el contador del programa se 
decrementa en 2, y la instrucción vuelve a ejecutarse. 


Flujo de datos: Е] am 
7777 
в СОМА € Ето == % а | 
ЖОО ый. 2-43 
Tiempo: В = 0; 4 ciclos M; 16 estados T; 8 seg @ 2 MHz. 
B 4 0: 5 ciclos M; 21 estados T; 10.5 useg @ 2 MHz. 
Direccionamiento: Externo. 
Banderas: 52 H ммм с 
Ejemplo: INIR 
Antes: Después: 
ULL s с Ә ЖШ зы |с 
[ 2 |рмвто ҰЙ PUERTO 
51 5] 
| ED Ca MA 
e ое. 
ae pu 
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JP cc, pq Salto condicional a la posición pq. 


Función: Si сс es cierto: РС +- ра 
Formato: | 1 ШЕЕ о |1 [o | byte 1 
[ 1 3-і byte 2: dirección, byte inferior 
[+ та -] byte 3: dirección, byte superior 
Descripción: Si se cumple la condición especificada, la dirección de dos bytes 


inmediatamente siguiente al código de operación se cargará en 
el contador del programa; el código de operación se carga en la 
parte inferior del PC. Si la condición no se satisface, se ignora 
la dirección; cc puede ser: 


NZ — 000 no cero 
Z — 001 cero 
NC — 010 sin acarreo 
С — 011 acarreo 
PO — 100 paridad impar 
РЕ - 101 paridad par 
P — 110 más 
M - 111 menos 


Flujo de datos: 


Tiempo: 3 ciclos M; 10 estados T; 5 useg @ 2MHz. 
Direccionamiento: Inmediato. 
Códigos byte: ' CC: NZ Z NC C PO PE P M 


те Тока T 
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Banderas: 


Ejemplo: 


DA 


3B 


CODIGO 
OBJETO 
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LI 


5 Z H PN М C 


O TI] (eo suo 


JP C,3B24 


Antes: Después: 
PC 0032 4 Л ТР, 


JP pq Salto a la posición ра. 


Función: PC — pq 


Formato: Br] Ей 5 ofo] sta Ш byte 1: C3 
t e —] byte 2: dirección, byte inferior 
== byte 3: dirección, byte superior 
E == AE AA 


Descripción: El contenido de la posición de' memoria inmediatamente si- 
guiente al código de operación se carga en la mitad inferior del 
contador del programa y el de la posición siguiente a la ante- 
rior en la mitad superior del mismo contador. La siguiente 
instrucción se toma de esta nueva dirección. 


Flujo de datos: 


Tiempo: 3 ciclos М; 10 estados T; 5 useg @ 2 MHz. 


Direccionamiento: Inmediato. 
Banderas: 5 z H Ру м с 


(efecto nulo) 


Ejemplo: JP 3025 
Antes: Después: 
es pe [Ts] к TA 
A ul 
CODIGO 


OBJETO 


JP (HL) 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


266 


Saltar a (HD). 


РС + HL 


000000000 


El contenido del раг de registros HL se carga en el contador 
del programa. La siguiente instrucción se toma de esta nueva 
dirección. 


РС 


1 ciclo M; 4 estados T; 2 useg @ 2 MHz. 


Implícito. 

S 2. Нн. РМ N C 

| | СІ Д | ЕА (efecto nulo) 
JP (HL) 

Antes: Después: 


н| 0411 L 
8001 с И MM УО 


JP (IX) Salto a (IX). 


Función: PC — IX 


Formato: 


Descripción: El contenido del registro IX se carga en el contador del progra- 
ma. La siguiente instrucción se toma de esta nueva dirección. 


Flujo de datos: 


осо» 
m 


EET 


ЭУ 


Tiempo: 2 ciclos M; 8 estados T; 4 useg @ 2 MHz. 
Direccionamiento: Implicito. 
Banderas: s z H 


m [ | | ИА (efecto nulo) 


Ejemplo: JP (IX) 
Antes: Después: 
> 
CODIGO 
OBJETO 
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JP (IY) Salto a (IY). 


Función: PC — IY 


RIP aaa bye LED 


БЦ byte 2: E9 


Descripción: El contenido del registro ТУ se lleva al contador del programa. 
La siguiente instrucción se toma de esta nueva dirección. 


Flujo de datos: 


ос» 
m 


Tiempo: 2 ciclos M; 8 estados T; 4 useg @ 2 MHz. 
Direccionamiento: Implícito. 
Banderas: 
5 7 H PV N C 
| | | il Г | | | (efecto nulo). 
Ejemplo: JP (IY) 
Antes: Después: 
—— SE 71 < 2% 
y) 
CODIGO 
OBJETO 
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JR cc,e 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Salto relativo condicional de longitud e. 


Si cc es cierto, РС — PC + е 


ETT bye à 
F A A 
LUI D 5 a a d 


-| byte 2: valor del desplazamiento 


Si se satisface la condición especificada, el valor del desplaza- 
miento dado se suma al contador del programa en aritmética de 
complemento a dos, para que el salto pueda darse hacia adelante 
o hacia atrás. El valor del desplazamiento se suma al valor de 
PC + 2 (después del salto), de manera que el desplazamiento 
efectivo es de — 126 a +129 bytes. El ensamblador resta auto- 
máticamente 2 del valor del desplazamiento fuente para generar 
el código hexadecimal. Si la condición no se satisface, se ignora el 
valor del desplazamiento y la ejecución de instrucciones prosigue 
en secuencia; cc puede ser: 


NZ — 00 NC — 10 
2-(01 с- 1 


г о о» 


LOGICA 
DE CONTROL! 


222222222222 


РС 


Ciclos М Estados Т @ MES 
condición 
cierta: 3 12 6 
condición 
falsa: 2 7 3.5 
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Direccionamiento: 


Códigos byte: 
Banderas: 


Ejemplo: 


CODIGO 
OBJETO 
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Relativo. 


сс NZ Z МС С 


5 z H PNN с 
(efecto nulo) 
JR NC,$ — 3 $ = PC actual 
Antes: Después: 
Го | Го |, 


JRe 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


Salto relativo de longitud e. 


PC<-PC+e 


0 о [о BENE 0 o| byte 1: 18 
byte 2: valor del desplazamiento 


El valor del desplazamiento dado se suma al contador del pro- 
grama en aritmética de complemento a 2, para que el salto 
pueda darse hacia adelante o hacia atrás. El valor del desplaza- 
miento se suma al valor de PC + 2 (después del salto), de 
manera que el desplazamiento efectivo es de — 126 a + 129 
bytes. El ensamblador resta automáticamente 2 del valor del 
desplazamiento fuente para generar el código hexadecimal. 


тоо > 


3 ciclos М; 12 estados T; 6 ¡seg @ 2 MHz. 


Relativo. 

SZ H рум с 
Fesses (efecto nulo) 
JR D4 

Antes: Después: 


B100 PC 


(Salto hacia atrás.) 


271 


LD dd, (nn) 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 
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Carga el par de registros dd a partir de las posiciones de 
memoria direccionadas por nn. 


dd;,, = (пп); dd,,, < (nn + 1) 


[г РР | byte 1: ED 
[o]: 44 |1 o ]1]1] byte 2 


byte 3: dirección, byte inferior 


byte 4: dirección, byte superior 


| 


Е contenido de la posición de memoria direccionada por la 
posición inmediatamente siguiente al código de operación se 
carga en la parte inferior del par de registros especificado. А 
continuación se carga el contenido de la posición de memoria 
siguiente a la anterior en la parte superior del mismo par de 
registros. El byte de orden inferior de la dirección nn sigue 
inmediatamente al código de operación; dd puede ser: 


BC — 00 HL — 10 
DE - 01 SP — 11 


6 ciclos M; 20 estados T; 10 useg @ 2 MHz 


Directo. 


Códigos byte: 
Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


dd: BC DE HL SP 


e [æ] se] ев] тв] 


5 2 H Ру N С 


ГІТІТТІТІ «воо nulo 


LD DE, (5021) 


Antes: Después: 


D DBE2 Je D ПРИ Е 


5022 5022 


sm| F4 | 5071 | м | 
[| 30 | 
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LD dd, nn Carga el par de registros dd con el dato inmediato nn. 


Función: dd — nn 


Formato: БЕК 0 ЭП 1 | byte 1 
AA byte 2: dato inmediato inferior 
byte 3: dato inmediato superior 


Descripción: Los contenidos de las dos posiciones de memoria inmediata- 
mente siguientes al código de operación se cargan en el par de 
registros especificado. El byte de orden inferior del dato aparece 
justo a continuación del código de operación; dd puede ser: 


BC — 00 HL — 10 
DE — 01 SP — 11 
Flujo de datos: A 
à 5 (йы 
=== 
н l 
p.m] 
Bu — Л 
Tiempo: 3 ciclos M; 10 estados T; 5 useg @ 2 MHz. 
Direccionamiento: Inmediato. 
Códigos byte: dd: BC 
Banderas: 5 2 H PN м с 
СІТІТТІТІ eese nuo 
Ejemplo: LD DE,4131 
Antes: Después: 
| omn | 
NEUE 
CODIGO 
OBJETO 
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LD r, n 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Códigos byte: 
Banderas: 


Ejemplo: 


| e | 
3B 
— 3 


CODIGO 
OBJETO 


Carga el registro r con el dato inmediato n. 


ren 


ee ааа byte 


byte 2: dato inmediato 


п — 


El contenido de la posición de memoria inmediatamente si- 
guiente al código de operación se carga en el registro especifica- 
do; r puede ser: 


А — 111 E — 011 
B — 000 H - 100 
C — 001 L — 101 
D — 010 


LD 


2 ciclos M; 7 estados T; 3.5 useg 2 MHz. 


тоор» 


CE 
E 
L 


Inmediato. 
ГА B C D E H L 

as [os [oe [16] ve [26] 7] 

$ 2 H рум с 

(efecto nulo) 

LD C,3B 

Antes: Después: 
со ] с 5% 
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LD гг 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Códigos byte: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 
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Carga el registro г con el contenido del г. 


El contenido del registro fuente especificado se carga en el 
registro destino especificado; r y r' pueden ser: 


A — 111 E — 011 
B — 000 H — 100 
C — 001 L — 101 
D — 010 

A 

B с 

D E 

H L 


1 ciclo M; 4 estados T; 2 шер @ 2 MHz. 
Implicito. 


A BC D EHL (fuente) 


5 7 H Рм м C 


| (efecto nulo) 
LD H,A 

Antes: Después: 
A A 
EE HE « ЕЙ 


LD (BC), А 


Función: 
Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


Carga la posición de memoria indirectamente direccionada (BC) 
a partir del acumulador. 


(BC) — A 


[eje[olefo[o[tr[oj 02 


El contenido del acumulador se carga en la posición de memo- 
ria direccionada por el contenido del par de registros BC. 


Las 


TOO p 


2 ciclos M; 7 estados Т; 3.5 изе; @ 2 MHz. 


Indirecto. 

5 2 H PAN C 
ГІТТІТІТІ ee nulo 
LD (BC)A 

Antes: Después: 


Ар” RE 
ао Je 


с в 
== 4109 шеші 
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LD (DE), А 


Función: 


Formato: 
Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 
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Carga la posición de memoria indirectamente direccionada (DE) 
a partir del acumulador. 


(DE) — A 


| Го То | оо | [0] 12 


El contenido del acumulador se carga en la posición de memo- 
ria direccionada por el contenido del par de registros DE. 


Ir O m» 


Ze 


LR 


2 ciclos M; 7 estados T; 3.5 изер @ 2 MHz. 


Indirecto. 
S Z H PNN C 
| | | (efecto nulo) 
LD (DE), А 
Antes: Después: 


LD (H L), n Carga el dato inmediato n en la posición de memoria indirecta- 
mente direccionada (HL). 


Función: (HL) — n 
Formato: byte 1: 36 

nt n +i byte 2: dato inmediato 
Descripción: El contenido de la posición de memoria inmediatamente si- 


guiente al código de operación se carga en la posición de 
memoria indirectamente direccionada por el apuntador HL. 


Flujo de datos: 


то» > 


Tiempo: 3 ciclos M; 10 estados T; 5 useg @ 2 MHz. 
Direccionamiento: Inmediato/indirecto. 
Banderas: 5 2 H Ру N с 

Reed es (efecto nulo) 
Ejemplo: LD (HL), 5A 

Antes: Después: 


EC p 
и | 
CODIGO 


OBJETO 


LD (H L), r Carga la posición de memoria indirectamente direccionada (HL) 
a partir del registro r. 


Función: (HL) «г 
puma; 00000225 
Descripción: El contenido del registro especificado se carga en la posición de 
memoria direccionada por el par de registros HL; r puede ser: 
A — 111 E — 011 
B — 000 H - 100 
C — 001 L — 101 
D — 010 
Flujo de datos: A 
B c 
D E 
H 2 


— 


Tiempo: 2 ciclos M; 7 estados T; 3.5 useg @ 2 MHz. 
Direccionamiento: Indirecto: 

Códigos byte: nr AB C DEH 

Banderas: $ z H Ру N C 


(efecto nulo) 
Ejemplo: LD (HL), B 


Antes: Después: 


CODIGO 
OBJETO 
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LD r, (IX T d) Carga el registro r indirecto a partir de la posición de memoria 
indexada (IX + d). 


Función: ге 1Х + d) 


0000000002 
CET] ve? 


n d e| byte 3: valor del desplazamiento 


Formato: 


Descripción: El contenido de la posición de memoria direccionada por el 
registro de índice IX más el valor del desplazamiento dado se 
carga en el registro especificado; r puede ser: 


A — 111 E — 011 
B — 000 H — 100 
C — 001 L — 101 
D — 010 


Flujo de datos: 


Tiempo: 5 ciclos M; 19 estados T; 9.5 useg @ 2 MHz. 
Direccionamiento: Indexado. 


Códigos byte: = À В С D Е 


H L 
ov- [e| e e [se se [06] e] « 


S 7 H P/V N C 


| | | | | | (efecto nulo) 


Banderas: 
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Ejemplo: LD E, (IX + 5) 
Antes: Después: 
Ге Je ZZ 
о | 


Го | 3020 | — 2A 3020 

= ET eR 
LE 3025 3025 | 15 

CODIGO | ы 


OBJETO 
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LD r, (IY + d) Carga el registro indirecto r a partir de la posición de memoria 
indexada (IY + d). 


Función: ге (IY + d) 


ПЕСО | byte 1: FD 
0|! | | o | byte 2 
byte 3: valor del desplazamiento 


Formato: 


Descripción: El contenido de la posición de memoria direccionada por el 
registro de índice IY más el valor del desplazamiento dado se 
carga en el registro especificado; r puede ser: 


А — 111 E — 011 
B — 000 H — 100 
C — 001 L — 101 
D — 010 


Flujo de datos: 


Tiempo: 5 ciclos M; 19 estados T; 9.5 useg @ 2 MHz. 
Direccionamiento: Indexado. 


Códigos byte: 


r A B C D E Н L 
о [26 [46| & | se | se| во] в |. d 


Banderas: 
5 7 H PNN С 


(efecto nulo) 
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Ejemplo: 


| o | 
чш] 


CODIGO 
OBJETO 
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LD A,(IY + 2) 


Antes: 


Después: 


LD (IX + d),n 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 


Carga la posición de memoria indexada (IX + d) con el dato 
inmediato n. 


(IX +d)<n 


il efofifofrjo]+] byte 1: DD 
o] of +] +[o[+[+[o] byte 2: 36 
byte 3: valor del desplazamiento 


gb e — -] byte 4: dato inmediato 


Ll 1 1 bl 


El contenido de la posición de memoria que sigue inmediata- 
mente al valor del desplazamiento se transfiere a la posición de 
memoria direcccionada por el contenido del registro de índice 
más el valor del desplazamiento. 


I o wp 
m 


[с ы 
ЖИЫ 
е к 
ат 


5 ciclos M; 19 estados T; 9.5 useg @ 2 MHz. 


Indexado/inmediato. 


Г Er ва [| | ] Ж (efecto nulo) 


285 


Ejemplo: LD (IX + 4, FF 


Antes Después 
xL ве EE] 

Се | вю | во & | 
== ES КӨШ 
| o | и 
ME B10D[ ж | во 
| y] p] 

CODIGO 

OBJETO 
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LD (IY + а), п 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 


Carga la posición de memoria indexada (ГУ + а) con el dato 
inmediato n. 


(IY +d)<n 


E | о byte 1: FD 


о [о [|1 јо (1 [0] byte 2: 36 
byte 3: valor del desplazamiento 
byte 4: dato inmediato 


El contenido de la posición de memoria que sigue inmediata- 
mente al valor del desplazamiento se transfiere a la posición de 
memoria direccionada por el registro de índice más el valor del 
desplazamiento. 


E = шщ 


оо» 


IY 


p 27 


5 ciclos M; 19 estados T; 9.5 useg @ 2 MHz. 
Indexado/inmediato. 


s z н PNN с 
(efecto nulo) 
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Ejemplo: LD (IY + 3, BA 


Ántes: Después: 


CODIGO 
OBJETO 
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LD (IX + d), г 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Códigos byte: 


Banderas: 


Carga la posición de memoria indexada (IX + d) a partir del 
registro r. 


(IX + 4 er 


@ ЕЕ ШЕ +] byte 1: DD 
БТЕ] әуе? 
[0 A] byte 3: valor del desplazamiento 


El contenido del registro especificado se carga en la posición de 
memoria direccionada por el contenido del registro de índice 
más el valor del desplazamiento dado; r puede ser: 


A — 111 E — 011 
B — 000 H — 100 
C — 001 L — 101 
D — 010 


Ж 


2 


5 


5 ciclos M; 19 estados Т; 9.5 useg @ 2 MHz. 


Indexado. 
T: АВ C D E H L 
[= о р [4-3 


2 H РУ М C 
(efecto nulo) 
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Ejemplo: 


CODIGO 
OBJETO 
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4462 


LD (IX + 1),C 


Antes: 


LD (IY + а), ғ 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Códigos byte: 


b 


Banderas: 


Carga la posición de memoria indexada (IY + d) a partir del 
registro r. 


(IY + 4) -т 


1 | 1 1 Ц 1 110 |1 byte 1: FD 


ШШШ ы byte 2 
byte 3: valor del desplazamiento 


El contenido del registro especificado se carga en la posición de 
memoria direccionada por el contenido del registro de índice 
más el valor del desplazamiento dado; r puede ser: 


A — 111 E — 011 
B — 000 H — 100 
C — 001 L — 101 
D — 010 


5 ciclos M; 19 estados T; 9,5 useg @ 2 MHz. 


Indexado. 


г: А 


B 
MCICICICICICICIE 


ШЕ a T] (efecto nulo) 
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Ejemplo: 


FD 
77 


CODIGO 
OBJETO 


202 


5АВ4 


5AB7 


LD (IY + 3),А 


Antes: 


зар? ZE 
Lj 


LD A, (nn) 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Carga el acumulador a partir de la posición de memoria (nn). 


A < (mn) 
ПИН 1 pep DE [o] byte 1: 3A 


byte 2: dirección, byte inferior 


E | byte 3: dirección, byte superior 


El contenido de la posición de memoria direccionada por el 
contenido de las dos posiciones de memoria que siguen al 
código de operación se cargan en el acumulador. El byte de 
orden inferior de la dirección aparece justo a continuación del 
código de operación. 


тоо>» 
- о 


4 ciclos M; 13 estados Т; 6.5 useg @ 2 MHz. 
Directo. 


S 2 H Рум с 
(efecto nulo) 
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Ejemplo: 


CODIGO 
OBJETO 
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LD A.(3301) 


Antes: 


Después: 


LD (nn), A 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 


Carga la posición de memoria directamente direccionada (nn) a 
partir del acumulador. 


(nn) + А 


[o o|+]+J[o]o[+[o] byte 1: 2 
[———— ALCOR O тела ГТ byte 2: dirección, byte inferior 
TT 1 - byte 3: dirección, byte superior 


El contenido del acumulador se carga en la posición de memo- 
ria direccionada por el contenido de las posiciones de memoria 
que siguen al código de operación. El byte inferior de la direc- 
ción aparece justo a continuación del código de operación. 


IOD > 


4 ciclos M; 13 estados T; 6.5 useg @ 2 MHz. 
Directo. - 


S 2 H Ру М C 


CLLLLLLI] teo шо) 
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Ejemplo: 


CODIGO 
OBJETO 


296 


LD (0321, А 


Antes Después: 


N 


032 [ 06 | 0921 Mes 


LD (nn), dd 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Códigos byte: 


Carga las posiciones de memoria direccionadas por nn a partir 
del par de registros dd. 


(nn) $n айу; (nn + 1) ER dd,» 


[ПО] byte 1: ED 
MO В 


[—————— byte 3: dirección, byte inferior 


byte 4: dirección, byte superior 


El contenido de orden inferior del par de registros especificado 
se carga en la posición de memoria direccionada por las. posi- 
ciones de memoria que siguen al código de operación. El conte- 
nido de orden superior del par de registros se carga en la 
posición de memoria que sigue a la cargada a partir del orden 
inferior. El orden inferior de la dirección nn aparece justo a 
continuación del código de operación; dd puede ser: 


BC — 00 HL — 10 
DE — 01 SP — 11 


ос» 


—<Ҷ 
2222 


6 ciclos M; 20 estados T; 10 seg @ 2 MHz. 


Directo. 


dd: BC DE HL SP 


Зоесла 


297 


Banderas: 5 z H PV N C 
(efecto nulo) 


Ejemplo: LD (040B), BC 


Antes: Después: 


оп К ош | 


048 | 06 | 0408 ЙЙЙЙ 
040€ 040C E 


CODIGO 
OBJETO 


298 


LD (nn), HL 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 


Carga las posiciones de memoria direccionadas por nn a partir 
de HL. 


(nn) € E; (nn + 1) < H 


ШЕ Е pa pere o] byte 1: 22 
Е---- byte 2: dirección, byte inferior 


Е--- A Е byte 3: dirección, byte superior 


El contenido del registro L se carga en la posición de memoria 
direccionada por las posiciones de memoria que siguen al código 
de operación. El contenido del registro H se carga en la posi- 
ción de memoria que sigue a la anterior. El byte inferior de la 
dirección nn aparece justo a continuación del código de opera- 
ción. 


TOU» 
mo 


5 ciclos M; 16 estados Т; 8 pseg @ 2 MHz 
Directo. 


рум с 


ГТІТТІТТІ ee malo 


Ejemplo: 


Гы | 
ETE 


CODIGO 
OBJETO 


300 


LD (40B9) HL 


Antes: 


Después: 


(C | 


4069 АТ 
«вл % 


LD (nn), IX 


Función: 


Formato: 


| Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


FERE 


Carga las posiciones de memoria direccionadas por nn a partir 
de IX. 


(nn) та ТХ аз (nn + 1) es ТХ. 


СЕЕ byte 1: DD 
оо | 1 [о јо [о "То | byte 2: 22 


[————"————{| byte 3: dirección, byte inferior 


L 1 Жаа. 


[———”- — - | byte 4: dirección, byte superior 


El contenido de orden inferior del registro IX se carga en la 
posición de memoria direccionada por el contenido de memoria 
que sigue al código de operación. El contenido de orden supe- 
rior del registro IX se carga en la posición de memoria que 
sigue a la anterior. El byte inferior de la dirección nn aparece 
justo después del código de operación. 


тою р 
то 


ИЯ 


| 


6 ciclos М; 20 estados Т; 10 pseg @ 2 MHz. 


Directo. 


S Z H РУ М С 


(efecto nulo) 


301 


Ejemplo: 


CODIGO 
OBJETO 


302 


LD (012B) IX 


Antes: 


0406 


Después: 


0128 ГА 
012c ж 


| LD ( nn), IY Carga las posiciones de memoria direccionadas por nn a partir 
| де ТҮ. 
{ 

Кипсїбп: (nn) < ТҮ; (nn + 1) — IY up 

Formato: 


ОРГ ГГ [Г] bue 1: FD 


olofilolololrlo byte 2: 22 


| —————8——— j| byte 3: dirección, byte inferior 


| 1 1 L 1 1 1 
byte 4: dirección, byte superior 
Descripción: El contenido de orden inferior del registro IY se carga en la 


posición de memoria direccionada por el contenido de la posi- 
ción de memoria que sigue al código de operación. El contenido 
de orden superior de IY se carga en la posición de memoria 
que sigue a la anterior. El byte inferior de la dirección nn 
aparece justo después del código de operación. 


Flujo de datos: 


A 

B с 

D E 

H 1 

Ветово №) 

Г 

Тіетро: 6 ciclos M; 20 estados T; 10 useg @ 2 MHz. 
Direccionamiento: Directo. 
Banderas: 


5 2 Н РМ м С 


ГІІТІТІ!! eto nuo 


Ejemplo: LD (BD04) IY 


Antes: Después: 


ГГ 
а БЕГЕН ын. 


CODIGO 
OBJETO 


LD A, (BC) 


Función: 


Formato: 
Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


Carga el acumulador a partir de la posición de memoria 
indirectamente direccionada por el par de registros BC. 


А < (BC) 


оо То о То То | ОА 


El contenido de la posición de memoria direccionada рог е! 
contenido del par de registros BC se carga en el acumulador. 


же 
кр 


2 ciclos M; 7 estados T; 3.5 шер. @ 2 MHz. 


TOO > 


Indirecto. 
S Z H PN N C 

№ | ТО) | ] | | (efecto nulo) 

LD А, (ВС) 

Antes: Después: 

A| ^e | АЙ 

в м |с | x je 
320] 4 | за м | 


LD A, (DE) 


Función: 


Formato: 
Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


Carga el acumulador a partir de la posición de memoria indi- 
rectamente direccionada por el par de registros DE. 


А < (DE) 


оо | ле falo] SC 


El contenido de la posición de memoria direccionada por el 
contenido del par de registros DE se carga en el acumulador. 


ГГА 


2 ciclos М; 7 estados T; 3.5 seg @ 2 MHz. 


Indirecto. 
5 72 H Рум C 
(efecto nulo) 
LD A,(DE) 
Antes: Después: 
A А ZA 


LD A, | Carga el acumulador a partir del registro vector de interrupcio- 
nes l. 


Función: А —I 


Formato: byte 1: ED 
БЕРБ byte 2: 57 


Descripción: El contenido del registro vector de interrupciones I se carga en 
el acumulador. 


Flujos de datos: 


A 
B с 
D E 
H с 
І 
Tiempo: 2 ciclos M; 9 estados T; 4.5 useg @ 2 MHz. 
Direccionamiento: Implícito. 
Banderas: s Z н орум с 
ejej lol хо | | 
Se activa según el contenido de IFF2. 
Ejemplo: LD A, 1 
Ántes: Después: 
A 30 il 4 | 22222 [ав] 


CODIGO 
OBJETO 


307 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


308 


Carga el registro vector de interrupciones I a partir del acumu- 
lador. 


I—A 


ВО] byte 1: ED 
“Е byte 2: 47 


El contenido del acumulador se carga en el registro vector de 
interrupciones. 


г Әә со» 


2 ciclos М; 9 estados T; 4.5 useg €? 2 MHz. 


Implícito. 
s z H PN М C 
ET | ша! | | (efecto nulo) 
LD LA 
Antes: Después: 
ele "шиша А 


LD A,R 


Función: 
Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 


Ejemplo: 


| O | 


CODIGO 
OBJETO 


Carga el acumulador a partir del registro de refresco de memo- 
ria R. 


А-В 


Ш epi РР | Бие 1: ЕО 
CE ER byte 2: 5Е 


El contenido del registro de refresco de memoria se carga en el 
acumulador. 


=з ш 


2 ciclos M; 9 estados T; 4.5 useg @ 2 MHz. 
Implícito. 


5 PN N C 


ee] T» | T«I5T | 


Se carga con el contenido de IFF2. 


LD A,R 
Antes: Después: 
A[ 62 в 4 ап с 4 1 


LD HL, (пп) 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 


310 


Carga el registro HL a partir de la posición de memoria direc- 
cionada por nn. 


L — (nn); H — (nn + 1) 


ng ШЕ ШЕ 1 Е | byte 1: 2А 
РТА РР TT byte 2: dirección, byte inferior 


T T a T T T . .. . 
— | byte 3: dirección, byte superior 


El contenido de la posición de memoria direccionada por las 
posiciones de memoria que siguen al código de operacón se 
cargan en el registro L. El contenido de la posición de memoria 
siguiente a la anterior se carga en el registro H. El byte inferior 
de la dirección nn aparece justo después del código de opera- 
ción. 


5 ciclos М; 16 estados T; 8 useg € 2 MHz. 


Directo. 


СІ i ada (efecto nulo) 


Ejemplo: 


CODIGO 
OBJETO 


LD HL, (0024) 
Antes: 


H ОВВЕ 


Después: 


н 322 «ос ИИД, 


ooza| 69 | 
| 4 | 


311 


LD iX, nn 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


312 


Carga el registro IX con el dato inmediato nn. 


IX — nn 


1 отт [оту byte 1: DD 


оі о | т |о [о [о 1 | byte 2: 21 


i — - byte 3: dato inmediato, byte inferior 
€—— - byte 4: dato inmediato, byte superior 


El contenido de la posición de memoria que sigue al código de 
operación se carga en el registro IX. El byte inferior aparece 
justo después del código de operación. 


ПИТ 


4 ciclos M; 14 estados T; 7 ¡seg @ 2 MHz. 


Inmediato. 
S 2 H рум с 
[TITII LI) eto nu) 
LD IX, BOB1 
Antes: У 
306F аа 7, 


LD IX, (nn) 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Carga el registro IX a partir 
direcciondas por nn. 


IX;,, < (nn); [Хр — (nn + 1) 


ШЕГЕ [1] byte 
БЫТ 51-5180) һе 


de las posiciones de memoria 


1: DD 


кә 


: 2А 
3: dirección, byte inferior 


4: dirección, byte superior 


El contenido de la posición de memoria direccionada por las 
posiciones de memoria que siguen al código de operación se 
carga en el byte inferior del registro 1X. El contenido de la 
posición de memoria que sigue a la anterior se carga en el byte 
superior de IX. El byte inferior de la dirección nn aparece justo 
después del código de operación. 


тоо> 


ЕЕ 


x K KL LLRA 


6 ciclos M; 20 estados Т; 10 useg @ 2 MHz. 


Directo. 


PH N С 


(efecto nulo) 


Ejemplo: LD IX, (010B) 


Antes: Después: 
x на ] Жж 
| 0D | оо — | oo| | 
оюс[ | oc за | 
| o | 
ES 
CODIGO 
OBJETO 


` 314 


LD IY, nn Carga el registro IY con el dato inmediato nn. 


Función: IY -nn 


Formato: pala 1 То byte 1: FD 


ЕЕ ШЕ ee 


[2 byte 3: dato inmediato, byte inferior 
Su x T ГТ . 3 B 
—— +) byte 4: dato inmediato, byte superior 
Descripción: El contenido de la posición de memoria que sigue al código de 


operación se carga en el registro IY. El byte inferior aparece 
justo después del código de operación. 


Flujo de datos: 


Әә ә p 


Tiempo: 4 ciclos M; 14 estados Т; 7 изе; @ 2 MHz. 
Direccionamiento: Inmediato. 
Banderas: $ 2 H PV N с 

LELE LI) (teo mio 
Ejemplo: LD IY,21 

Antes: Después: 


м 0698 A ZZ AGA 


CODIGO 
OBJETO 


LD IY, (nn) Carga el registro IY a partir de la posición de memoria direc- 
cionada por nn. 


Función: IY; (nn); TY,,, — (nn + 1) 


РРРРЪГРЙ | byte 1: FD 
(191316 ШЕ [4 [о] byte 2: 2А 
тото byte 3: dirección, byte inferior 


Formato: 


VA] Я фа п 
| а ти byte 4: dirección, byte superior 
Descripción: El contenido de la posición de memoria direccionada por las 


posiciones de memoria que siguen al código de operación se 
carga en el byte inferior del registro IY. El contenido de la 
posición de memoria que sigue a la anterior se carga en el byte 
superior del registro ТУ. El byte inferior de la dirección nn 
aparece justo después del código de operación. 


== 


с 
М 


LE 


Flujo de datos: 


то» > 


Tiempo: 6 ciclos M; 20 estados T; 10 изе; @ 2 MHz. 
Direccionamiento: Directo. 
Banderas: 


LT | | | | E (efecto nulo) 


316 


Ejemplo: LD ТУ, (500D) 


Antes: Después: 
IY ЖУУ, 


5000 09 | soo| — 0 __ 


500€ 


Е 


CODIGO 
OBJETO 


LD В, А Carga el registro de refresco de memoria R a partir del acumu- 
lador. 


Función: R+A 


Formato: "m JOE 0 1] byte 1: ED 
СООО byte 2: 4F 


Descripción: El contenido del acumulador se carga en el registro de refresco 
de memoria. 


Flujo de datos: 


A 

B 

D 

H 
Tiempo: 2 ciclos M; 9 estados T; 4.5 изе; @ 2 MHz. 
Direccionamiento: Implícito. 
Banderas: 5 z H P N с 

LII] ] (efecto nulo) 
Ejemplo: LD КА 
Antes: Después: 


O > — ^ ow Го) A EZ 


CODIGO 
OBJETO 


318 


LD SP, HL 


Función: 


Formato: 
Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


Cargar el puntero de la pila a partir de HL. 


SP < HL 


ШСБ] ro 


El contenido del par de registros HL se carga en el puntero de la 


pila. i 


в с 
s m — 
1 ciclo M; 6 estados T; 3 useg @ 2 MHz. 
Implícito. 
PN N c 


s z H 
ІСІ ШЕ [351] (efecto nulo) 


LD SP,HL 


Antes: Después: 


H | н овде t 
se 2 ” а 


319 


LD SP, IX Carga el puntero de la pila a partir del registro IX. 


Función: SP — IX 


dis ШЕЕ ЕК ПЕД зр, 
РСС] вие 2: F9 


Descripción: El contenido del registro IX se carga en el puntero de la pila. 
Flujo de datos: " 
B с 
D E 
H L 
Іх | 
SP 
Tiempo: 2 ciclos М; 10 estados T; 5 изе; @ 2 MHz. 
Direccionamiento: Implicito. 
Banderas: 5 z H РМ м С 
Lu] LI ] (efecto nulo) 
Ejemplo: LD SP,IX 
Antes: Después: 
O “4 IX 09D2 IX 0902 
EN | ; 
E SP 54AO бе LL 
OBJETO 
CODIGO 


320 


LD SP, ІҮ Carga el puntero de la pila a partir del registro ТУ. 


Función: SP — IY 


5 СТТ) bye 1: FD 
ВОО] byte 2: ro 


Descripción: El contenido del registro ТУ se carga en el puntero de la pila. 


Flujo de datos: 


тое > 


С 
Е 
L 
ІҮ 
5Р 


Tiempo: 2 ciclos M; 10 estados T; 5 useg @ 2 MHz. 
Direccionamiento: Implicito. 
Banderas: $ 2 H ру м C 


Ejemplo: LD SP, IY 


CODIGO 
OBJETO 


321 


LDD 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


322 


Carga de bloque con decremento. 


(DE) - (HL; DE + DE - 1; НЫ < HL — 1; ВС < BC- 1 


ео] byte 1: ED 
[S Jofrfo[r[ofofo] byte 2: A8 


El contenido de la posición de memoria direccionada por HL se 
carga en la posición de memoria direccionada por DE. А conti- 
nuación se decrementan BC, DE y HL. 


C 


EE 
в n 
d, DESTINO p 
ГАЛАП 


4 ciclos M; 16 estados T; 8 useg @ 2 MHz. 
Indirecto. 


Е 


а Se pone а 0 si ВС - 0 tras la ejecución; 
se activa a 1 en caso contrario. 


Ejemplo: 


АВ 


CODIGO 
OBJETO 


Después: 


B 


D E 


L 


йке 
2102222 
РНИИ 


LDDR 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


324 


Carga de bloque con decremento repetida. 


(DE) = (HL); DE + DE — 1; HL < HL - 1; 
ВС — BC — 1; repetir hasta que ВС = 0 


рее byte 1: ED 
о [5 о [о [о | byte 2: B8 


El contenido de la posición de memoria direccionada por HL se 
carga en la posición de memoria direccionada por DE. A conti- 
nuación se decrementan DE, HL y BC. Si BC + 0, el contador 
del programa se decrementa en 2, y la instrucción vuelve a 
ejecutarse. 


^p 
4 


HZ 


ч1----------- 


к-г 


ВС + 0: 5 ciclos М; 21 estados T; 10.5 изев @ 2 MHz. 
ВС = 0: 4 ciclos М; 16 estados Т; 8 useg @ 2 MHz. 


Indirecto. 


Sh. 17 H P/V N C 


Ejemplo: 


CODIGO 
OBJETO 


Después: 


8 
D 
H 


Г ЛА 
баб 
77777 


5 


P 


E 
L 


325 


LDI 
Función: 
Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


326 


. (DE) + (HL); DE + DE + 1; HL + HL + 1; BC < ВС – 1 


Carga de bloque con incremento. 


РР [iiel] byte 1: ED 
[ео [оо] о] о] byte 2: А0 


El contenido de la posición de memoria direccionada por HL se 
carga en la posición de memoria direccionada por DE. A conti- 
nuación se incrementan DE y HL y se decrementa el par de 
registros BC. 


8” 

в с 
ко m. 
н СА 


4 ciclos М; 16 estados T; 8 useg o 2 MHz. 

Indirecto. 

sz н PNN C 

I| fol [xie] | 
Se reinicia a O si BC = 0 tras la ejecución; 
se activa a 1 en caso contrario. 


Ejemplo: LDI 
Antes: 
0» | 
p — TE 
He | 

- | 

CODIGO 
зо | 


B 
D 


Después: 


7 2 а Ж 
н 72% 7722. 


за Ж 
и | 


3902 


327 


LDIR 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


328 


Carga de bloque con incremento repetida. 


(DE) — (HL); DE — DE + 1; HL — HL + 1; 
BC — BC — 1; repetir hasta que BC = 0 


Ар} byte 1: ED 
byte 2: 80 


El contenido de la posición de memoria direccionada por HL se 
carga en la posición de memoria direccionada por DE. A conti- 
nuación se incrementan DE y HL y se decrementa BC. Si 
BC + 0, el contador del programa se decrementa en 2, y la 
instrucción vuelve a ejecutarse. 


o 


дана „Е БАЦОВ 


D 
H 


c mar V 9 
Жаз 


— = а іт. 
ARS 


Si ВС +0: 5 ciclos M; 21 estados T; 10.5 seg @ 2 MHz. 
Si BC = 0: 4 ciclos M; 16 estados T; 8 иер @ 2 MHz. 


Indirecto. 


P/V N 


Ejemplo: 


CODIGO 
OBJETO 


ње 


Después: 


(4С 
Е 
ZZ 


% 
ж» 
| AA | 


329 


LD r, (H L) Carga el registro r indirecto a partir de la posición de memoria 


(HL). 
Función: r (HL) 
Formato: НО ее ШЕП 
Descripción: | El contenido de la posición de memoria direccionada por HL зе 
carga en el registro especificado; r puede ser: 
A — 111 E — 011 
B — 000 H — 100 
C — 001 L — 101 
D — 010 
Flujo de datos: A 
B с 
D E Gks 
н 
Tiempo: 2 ciclos М; 7 estados T; 3.5 useg à 2 MHz. 
Direccionamiento: Indirecto." 
Códigos byte: r A BCDE H 1 
[| eel | sef se] so] ce] 
Banderas: s z H PNN C 
СТО eao nulo 
Ejemplo: LD D,(HL) 
Antes: Después: 
D 2 % 
[бую]. «Loc Ta | 
oca | оса аа | 
~ E) 
CODIGO 
OBJETO 


330 


NEG 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


Negativiza el acumulador. 


A<0-A 


Pr] +fo[+[+[o[+] byte 1: ED 
Го | ТоТоТо | [|] byte 2: 44 


El contenido del acumulador se resta de cero (en complemento 
a dos), y el resultado vuelve a almacenarse en el acumulador. 


А 
[| 


2 ciclos M; 8 estados T; 4 useg @ 2 MHz. 


Implicito. 


5 z н ®WN с 

Фе |ө| joel е 

C se activa a 1 si A era 0 antes de la instrucción. 
P se activa a 1 si A era 80H. 


NEG 
Antes: Después: 


331 


NOP 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


332 


No opera. 


Retardo. 


olololololololol 00 


No ocurre nada durante un ciclo M. 


No actúa. 
с 
Е 
L 


тое > 


1 ciclo М; 4 estados T; 2 useg @ 2 MHz. 
Implicito. 


So zZ H PNN C 


СТГ) eue nulo 


OR $ 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


(HL) 


(IX + d) 


(IY + d) 


O lógica al acumulador y el operando s. 


А — Avs 


s puede ser r, n, (HL), (IX + d) o (IY + d). 


byte 


ЕЛЕНЕ Банкі 
ШЕШЕГЕ 


Буіе 


byte 


byte 
byte 
byte 


byte 


E 0| 1|! JE 1 o] byte 
byte 
r puede ser: 
A — 111 
B — 000 
C — 001 
D — 010 


: F6 

: dato inmediato 

: B6 

: DD 

: B6 

: valor del desplazamiento 
: FD 

: B6 


: valor del desplazamiento 


E — 011 
H — 100 
L — 101 


El acumulador y el operando especificado se someten a la 
operación O lógica, y el resultado se almacena en el acumula- 
dor; s se define en la descripción de instrucciones ADD simila- 


= 


es. 


$3 


333 


Tiempo: 


Direccionamiento: 


Códigos byte: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


334 


5 Ciclos М Estados T @ о Hz 
r 1 4 4 

n 2 7 3.5 
(HL) 2 7 3.5 

(IX + d) 5 19 9.5 

(IY + d) 5 19 9.5 


r: implícito; n: inmediato; (HL): indirecto; (IX + d), (IY + d): 
indexado. 


OR r r: A B C D E H L 
[57 | во [81 [во [es | ва [ss] 
5 z H Фу м C 

¡[ejej ¡o! jejojo| 

OR B 


Antes: Después: 


imm 


OTDR 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Salida de bloque con decremento. 


(C) - (HL); B < B — 1; HL < HL — 1; repetir hasta que В = 0. 


ее] byte 1: ED 
(ор р] byte 2: ВВ 


El contenido de la posición de memoria direccionada por el par 
de registros HL se lleva a la salida del dispositivo periférico 
direccionado por el contenido del registro C. Tanto el registro 
B como el par HL se decrementan. Si B +0, el contador del 
programa se decrementa en 2, y la instrucción vuelve a ejecutar- 
se. C proporciona los bits А0 a A7 del bus de direcciones; B 
proporciona, tras decremento, los bits A8 а А15. 


ша 


PUERTO 


AE 
ОН |< 


ES БЕНЕН 


H і 


В = 0: 4 ciclos M; 16 estados T; 8 useg @ 2 MHz. 
B +0: 5 ciclos M; 21 estados Т; 10.5 useg @ 2 MHz. 


Externo. 


H P/V N C 


S Z 
EME 


Ejemplo: 


CODIGO 
OBJETO 


336 


OTDR 
Antes 
Во | = | 
H 0051 i 
PUERTO 
ES 
004Е 
you 
0051 
ESS 


Después: 


в ОО Es lc 
Ж PUERTO 


ES 


A e cm 


io al 


OTIR 
Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Salida de bloque con incremento. 


(С) — (HL); В — B — 1; HL < HL + 1; repetir hasta que В 


ТОТО о 101 byte 1: ED 
lol [ РН [+] byte 2: вз 


=0 


El contenido de la posición de memoria direccionada por el par 
de registros HL se lleva a la salida del dispositivo periférico 
direccionado por el contenido del registro C. El registro B se 
decrementa y el par HL se incrementa. Si B + 0, el contador 
del programa se decrementa en 2, y la instrucción vuelve a 


ejecutarse. C proporciona los bits А0 a A7; В proporciona, 
decremento, los bits A8 a А15. 


A 
B 


PUERTO 


о м — 


В = 0: 4 ciclos М; 16 estados T; 8 useg @ 2 MHz. 
В + 0: 5 ciclos M; 21 estados T; 10.5 useg @ 2 MHz. 


Externo. 


tras: 


337 


Ejemplo: 


3 


CODIGO 
OBJETO 


338 


OTIR 
Antes 
o T] 
H 5550 
АО 
5550 | 68 | 
sssi| 02 | 
5552 
5553 | es | 
yA] 


Después: 


222 ^o jc 
22ГО 
SRA PUERTO 


5550 
5551 02 
5552 9А 


58| 6 | 


AO 


OUT (C), r 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 
Códigos byte: 
Ejemplo: 


CODIGO 
OBJETO 


Salida del registro r al puerto C. 


(C) er 


E 1 Jopi 0 | byte 1: ED 
Б сае ere 


El contenido del registro especificado se lleva al dispositivo peri- 
férico direccionado por el contenido del registro C; r puede ser: 


А — 111 
B — 000 
C — 001 
D — 010 


E — 011 
H — 100 
L — 101 


El registro C proporciona los bits А0 a A7 del bus de direccio- 
nes; el registro B proporciona los bits A8 a AIS. 


PUERTO 


то о > 


Е 
L 


3 ciclos М; 12 estados T; 6 useg @ 2 MHz. 


Externo. 


S Z H РУ М с 
Ж [| 1] L | (efecto nulo) 


r А B C D E H L 
ю- [ә] а | е] ар El 
OUT (С), B 
Antes: 
э» [пт К 


PUERTO 


Fl 


Después: 
го я (с 
22 PUERTO 


F1 


339 


OUT (№), А 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


Salida del acumulador al puerto М. 


| byte 1: D3 
-| byte 2: dirección puerto 


ЕЕЕ ШЕ 


| 


El contenido del acumulador se lleva al dispositivo periférico di- 
reccionado por el contenido de la posición de memoria que sigue 
al código de operación. 


3 ciclos M; 11 estados T; 5.5 useg @ 2 MHz. 


Externo. 

E LI Е (efecto nulo) 
OUT (0A), A 

Antes: Después: 


А5] [FE риєвто АГ я 1 EZ tno 
OA OA 


OUTD 


Función: 


Formato: . 


Descripción: 


Flujo de datos:, 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


Salida con decremento. 


(С) < (HL; BC - B- 1; HL- HL - 1 


EBTTSETTST bre 1: ED 


1 то | 1 | от 1 11 | byte 2: AB 


El contenido de la posición de memoria direccionada por е! par 
de registros HL se lleva al dispositivo periférico direccionado por 
el contenido del registro C. El registro B y el par HL se decre- 
mentan. C proporciona los bits А0 a A7 del bus de direcciones; В 
proporciona, tras decremento, los bits A8 a А15. 


С 


ВЕ 
500502 БЕНЕН 


В 


s z H N C 
CL [>] ІШЕ | ] Se activa a 1 si В = 0 tras la ejecu- 
| | ción; en caso contrario se reinicia a 0. 


OUTD 
Antes: Después: 
в c B c 
H і н 
PUERTO PUERTO 


341 


OUTI 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


^ Tiempo: 
Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


342 


Salida con incremento. 


(C) (Н; B -B — 1; HL- HL + 1 


ОРРРРР | byte 1: ED 
[°[[°[°[°[['] byte 2: АЗ 


El contenido de la posición de memoria direccionada por el par 
de registro HL se lleva al dispositivo periférico direccionado por 
el contenido del registro C. El registro B se decrementa y el par 
HL se incrementa. C proporciona los bits А0 a A7 del bus de 
direcciones; B, tras decremento, proporciona los A8 a A15. 


B 


PUERTO 


с 
Е 
L 


CONTADOR] | 
ONTADOR 
4 ciclos M; 16 estados T; 8 useg @ 2 MHz. 


H 


Externo. 
S Z H мм N с 
Е ? ‚|1 Se activa a 1 si B=0 tras la ejecución; 
en caso contrario se reinicia a 0. 
OUTI 
Ántes Después: 


ЭУ ee |с 


(Сол } Э УЖЖ 
PUERTO ЖАЙ PUERTO 
BB BB 
PRE 


POP аа 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Códigos byte: 


аа: 


Banderas: 


Extrae de la pila el par de registros qq. 


94у = (SP); qq, (SP + 1); SP < SP + 2 


L 


El contenido de la posición de memoria direccionada por el 
puntero de la pila se carga en el byte inferior del par de registros 
especificado, y a continuación se incrementa el puntero. El conte- 
nido de la posición de memoria que ahora direcciona el puntero 
se carga en el byte superior del par de registros, y el puntero 
vuelve a incrementarse; qq puede ser: 


BC — 00 HL — 10 
DE — 01 AF — 11 
A F 
B c 
D E 
H 1 


ev 


3 ciclos M; 10 estados T; 5 useg @ 2 MHz. 


Indirecto. 


BC DE HL AF 


[о [e [n 


5 7 H Ру М с 


LLCS eto ші 


Ejemplo: 


-- 1 


CODIGO 
OBJETO 


| 


0158 
oise 


Después: 


22Р 


0158 
015С 
0150р 


о 


POP IX 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Extrae de la pila el registro IX. 


ЕЕ ЕЕ ЕГ] bre 1: DD 
TE т [о о [о [о | byte 2: El 


El contenido de la posición de memoria direccionada por el 
puntero de la pila se carga en el byte inferior del registro IX, y se 
incrementa el puntero. El contenido de la posición de memoria 
que ahora direcciona el puntero se carga en el byte superior del 
registro IX, y el puntero vuelve a incrementarse. 


== 
| 
2222 


РРР 


гоо» 


ЅР 


4 ciclos М; 14 estados Т; 7 useg @ 2 MHz. 
Indirecto. 


Su ad H PV N C 


ЕТТ eo nuo 


345. 


Ejemplo: 


| oo | 


1 


CODIGO 
OBJETO 


Después: 


х %%ҰЙ 
» е 


POP IY 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Extrae de la pila el registro IY. 


ІҮ, < (SP); IY,, < (SP + 1); SP + SP + 2 


DUUHUHUNDUNM 1: FD 
ШОР] byte 2: Bt 


El contenido de la posición de memoria direccionada por el 
puntero de la pila se carga en el byte de orden inferior del regis- 
tro ГУ, y a continuación se incrementa el puntero. El contenido 
de la posición de memoria que ahora direcciona el puntero se 
carga en el byte superior del registro ГУ, y el puntero vuelve a in- 
crementarse. 


A 
B 
D E 
H 


4 ciclos M; 14 estados T; 7 useg @ 2 MHz. 
Indirecto. 


5 2 H PN М С 


СЕТИ ОТО ee nue) 


347 


Ejemplo: POP IY 


Antes: Después: 

Moa] VEZ 

s 3004 ЯУ, 
En == е 
ET 3006 зе | o | 
CODIGO amu ш ри 


OBJETO 


PUSH qq Introduce el par de registros qq en la pila. 
Función: (SP — 1) < ачы»; (SP — 2) < дды; SP < SP — 2 


И 2 


Formato: 


Descripción: Se decrementa el puntero de la pila, y a continuación se carga 
el contenido del byte superior del par de registros especificado 
en la posición de memoria direccionada por el puntero de la 
pila. Este vuelve a decrementarse, y se carga el byte inferior del 
par de registros en la posición direccionada ahora por el pun- 
tero; qq puede ser: 


BC — 00 HL — 10 
DE — 01 AF — 11 


Flujo de datos: 


Tiempo: 3 ciclos M; 11 estados T; 6.5 useg @ 2 MHz. 
Direccionamiento: Indirecto. 


Códigos byte: 


99: BC DE HL AF 


Ге 


Banderas: 
5 z H PV N с 


С] | | | | (efecto nulo) 


349 


Ejemplo: PUSH DE 


Ta 


во oa] 
CODIGO v T 


OBJETO 


350 — 


Después: ` 


o[ ^ ww» je 
> 2 


PUSH IX 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Introduce IX en la pila. 


(SP — 1) IX; (SP — 2) < IX,5; SP < SP — 2 


sup? 


рр о] вие 1: DD 
ГССР] byte 2: ES 


Se decrementa el puntero de la pila y se carga el byte 
superior del registro IX en la posición de memoria direccionada 
por el puntero. Este vuelve a decrementarse, y el byte inferior 
de IX se carga en la posición de memoria direccionada ahora 
por el puntero. 


TOO > 
o 


LD 


2С 


4 ciclos М; 15 estados T; 7.5 useg @ 2 MHz. 
Indirecto. 


руым с 


CEL TII eee по 


351 


Ejemplo: PUSH IX 


Antes: Después: 
IX 04A2 IX | 04A2 
» 
Со | пас 
0095 
Ex] шин 
CODIGO EU 
OBJETO 


352 


PUSH IY 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 


Introduce IY en la pila. 


(SP — 1) < IY,,; (SP — 2) - IY; SP < SP — 2 


ШЕШЕНГЕ byte 1: ЕО 
ШЕ 1 оо То 1| byte 2: ES 


Se decrementa el puntero de la pila y se carga el byte 
superior del registro IY en la posición de memoria direccionada 
por el puntero. Este vuelve a decrementarse, y el byte inferior 
de IY se carga en la posición de memoria direccionada ahora 
por el puntero. 


3 ciclos M; 15 estados T; 7.5 seg @ 2 MHz. 
Indirecto. 


5 2 H рм NC 
(efecto nulo) 


353 


Ejemplo: 


CODIGO 
OBJETO 


354 


Después: 


ІҮ 90BF 


PAZ à G LZ 


00B4 
00B5 
00B6 


RES b,s Poner a 0 el bit b.del operando s. 


Función: $, = 0 
Formato: 5: 
r [1 [1 [0 [0 oļi byte 1: CB 
ase | Вие 
(HL) ['lrfolofifofi[r] byte 1: CB 


IX +d) Пор liioi} byte t: DD 
: СВ 


1 
2 
1 
2 
1 
2 
byte 3: valor del desplazamiento 
4 
1 
2 
3 
4 


[е е | 

Бе 

Е 
пу+9 CDDP) вие 

ШЕЕ ЕЕЕ Е 

@ ИЕ 


: FD 
: СВ 


: valor del desplazamiento 


0 — 000 4 — 100 
1 — 001 5 — 101 
2 — 010 6 — 110 
3 — 011 7 — 111 
r puede ser: 
А — 111 E — 011 
В — 000 H — 100 
C — 001 L — 101 
D — 010 
Descripción: El bit especificado de la posición determinada por s se pone a 


0; s se define en la descripción de instrucciones BIT similares. 


355 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Códigos byte: 


RES b, (HL) 
RES b,(IX + d) 
RES b,(IY + d) 


Banderas: 


356 


Қ = 
8 С ГА 
D E 
H L 

. изед 
8 | Ciclos M Estados T @ 2 MHz 
г 2 8 4 
(HL) 4 15 7.5 
(IX + d) 6 23 11.5 
(IY + d) 6 23 11.5 


r: implícito; (HL): indirecto; (IX + d), (IY + d): indexado. 


RES b,r 


„ АВ © D E Ht 
CB— 9 


| | ВЕ) 88 | 89 | ЗА | 8B вс [вр | 
2 |97 |90 |91 | 92 |93 |94 


DDCB — b 0 1 2 3 4 5 6 7 
| | | EJA | | 
FDCB — 86 | ВЕ | 96 | 9E | Аб | AE | Bo | BE 


5 2 H Рм М С 


[LL LLLLLL] detecto nuo 


Ejemplo: RES 1,H 


Antes: 
ES 3 
CB 
CODIGO 
OBJETO 


Después: 


357 


RET 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


358 


Indirecto. 
5 2 H рум C 
| | | | | | | | | (efecto nulo) 
RET 
Antes: Después: 
08B1 ec ZT 
» ғ? LA 
330| 2 | aoj a | 
[~] [~ | 


Retorno de subrutina. 


PC iy < (SP); PC,, < (SP + 1); SP < SP + 2 


sup 


D +[o[o[»ToJo[»] C9 


El contador del programa se extrae de la pila tal como se 
describe en las instrucciones POP. La siguiente instrucción se 
toma de la posición señalada por el PC. 


3 ciclos М; 10 estados T; 5 useg @ 2 MHz. 


RET cc Retorno condicional de subrutina. 


Función: Si cc es verdadero: РС, < (SP); PC,,, — (SP + 1);SP < SP + 2 


sup 


Formato: 


1 |---с<---|0|010 
I]. 


Descripción: Si se satisface la condición, el contenido del contador del pro- 
grama se extrae de la pila tal como se describe en las instruccio- 
nes POP. La siguiente instrucción se toma de la posición conte- 
nida en el PC. Si la condición no se satisface, la ejecución de 
instrucciones continüa en secuencia. 


Flujo de datos: 


LOGICA 
DE 


NZ — 000 PO — 100 


Z — 001 PE — 101 
NC — 010 P — 110 
C — 011 M - 111 
Tiempo: Condición satisfecha: 3 ciclos M; 11 estados Т; 6.5 useg @ 2 
MHz. 
Condición no satisfecha: 1 ciclo M; 5 estados T; 2.5 useg 2 
MHz. 
Direccionamiento: Indirecto. 
Códigos byte: CC: NZ Z NC C PO PE 


P М 
[cles] eo oe e [es [ro [re | 


Banderas: 


5 7 H рум с 
ГІТТІТІТІІ ete nuo) 


Ejemplo: RET NC 
Antes Después: 
г ; 
Pc PC 
Г 1] ТА 
шә „ч 
| o | 511 85 8511 8 
Бет”) esa asi в 
CODIGO Sl > 


OBJETO 


360 


RETI 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Retorno de interrupción. 


PC,, < (SP); PC,, <= (SP + 1); SP + SP + 2 


Epp pE ЕЙ byte 1: ED 
[o] "То | с | 1121913) byte 2: 4D 


El contador del programa se extrae de la pila tal como se 
describe en las instrucciones POP. Los dispositivos periféricos 
Zilog reconocen esta instrucción como el final de una rutina de 
servicio a periférico para controlar adecuadamente las priorida- 
des de interrupción internas. Para volver a habilitar las inte- 
rrupciones es preciso ejecutar una instrucción El antes de 
RETI. 


4 ciclos M; 14 estados T; 7 useg @ 2 MHz. 
Indirecto. 


S 2 H P/V N C 


ATT eue шо 


361 


Ejemplo: RETI 


Antes: Después: 
el em] SIISE 
әб оо ^] ОТУ 
yq) EÓ | 


CODIGO 
OBJETO 


RETN 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
“Direccionamiento: 


Banderas: 


Retorno de una interrupción no enmascarable, 


PC ins < (SP); PC,, < (SP + 1); SP SP + 2; 1FF1 + IFF2 


пп] byte 1: ED 
ое [о [оо] 1] byte 2: 45 


El contador del programa se extrae de la pila tal como se 
describe en las instrucciones POP. A continuación se copia en 
IFF1 el contenido de IFF2 (biestable de almacenamiento), para 
restaurar el estado de la bandera de interrupciones antes de una 
interrupción no enmascarable. 


4 ciclos M; 14 estados T; 7 useg @ 2 MHz. 
Indirecto. 


S Z Н црумс 


ETT] eue шо) 


363 


Ejemplo: 


CODIGO 
OBJETO 


364 


RETN 
Antes: 


pe [A 
> 


Después: 
РС ZX 
SP 


8B4C 
8B4D 9 


RL s 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


(HL) 


(IX + d) 


(IY + d) 


Rotación a la izquierda del operando s a través del acarreo. 


— 


: CB 


: СВ 


16 
DD 
CB 


: valor del desplazamiento 
: 16 


FD 
CB 


: valor del desplazamiento 


: 16 


E — 011 
Н - 100 
L — 101 


El contenido de la posición del operando especificado se despla- 
za un bit hacia la izquierda. El contenido de la bandera de 
acarreo se lleva al bit 0, y el del bit 7 a la mencionada bandera. 
El resultado final vuelve a almacenarse en la posición de parti- 
da; s se define en la descripción de instrucciones RLC similares. 


365 


Tiempo: m "s 
Ciclos M Estados T Hseg 


| @ 2 MHz 

r 2 8 4 

(HL) 4 15 7.5 

(IX +4) 6 23 11.5 

(IY + d) 6 23 11.5 
Direccionamiento: r: implícito; (HL): indirecto; (IX + d), (IY + d): indexado. 
Códigos byte: RL r 


Г: А B C D E H L 
сет 10] v o [ro 4] 8 


Banderas: $2 H Фуу м C 
С queda determinado por el bit 7 de la fuente. 
Ejemplo: RL E 
Antes: Después: 
[e | Е [wr] У 
E Се 1 WU; 
CODIGO 
OBJETO 


366 


ВІА 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


| Tiempo: 


| Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


d 


Direccionamiento: 


Rotación a la izquierda del acumulador a través de la bandera 
de acarreo. 


A 


БЕЗЕ Ar 


El contenido del acumulador se desplaza un bit hacia la izquier- 
da. El contenido de la bandera de acarreo раза al bit 0, y el del 
bit 7 a la mencionada bandera (rotación de 9 bits). 


1 ciclo M; 4 estados Т; 2 изев @ 2 MHz. 
Implicito. 


S Z Н Ру N С 
LI] [e] | [efe] 


C queda determinado por el bit 7 de A. 


Antes: Después: 


ГАРАГ му РА 


367 


RLCA 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


368 


Rotación a la izquierda del acumulador con copia al acarreo. 


с А 


го [о [о его ||. 07 


El contenido del acumulador зе rota un bit hacia la izquierda. 
El contenido original del bit 7 se lleva a la bandera de acarreo 
y al bit 0. 


SEN 


1 ciclo M; 4 estados T; 2 useg @ 2 MHz. 
Implicito. 


S Z PNN с 


ТТ ГЇ Le 


C queda determinado por el bit 7 de А. 


RLCA 
Antes: Después: 


Nota: Con excepción de las banderas, esta instrucción es idénti- | 
са а ВГС A. Se ha incluido para garantizar la compatibili- 
dad con el 8080. | 


RLC г 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Códigos byte: 


Banderas: 


Rotación a la izquierda del registro r con copia al acarreo. 


LI folfefofofr]1] byte 1: CB 
О о. byte 2 


El contenido del registro especificado se rota а la izquierda. Е 
contenido original del bit 7 se lleva a la bandera de acarreo y al 
bit 0; r puede ser: 


А- 111 E — 011 
B; 000 H — 100 
C — 001 L — 101 
D — 010 


2 ciclos M; 8 estados T; 4 seg @ 2 MHz. 
Implícito. 
= 


: A B C D E H L 
еы ее 


Обвовосас 


С queda determinado рог el Би 7 del registro fuente. 


Ejemplo: RLC B 


Antes: 
| са | 
[ ® | Се Ге | 
CODIGO 
OBJETO 


370 


Después: 


F 


RLC (HL) 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


Ejemplo: 


СВ 


CODIGO 
OBJETO 


Rotación a la izquierda de la posición de memoria (HL) con 
copia al acarreo. 


A 


(HL) 


[о 10 |10 [111] byte 1: CB 
ое оо а [119 [9| ук 22:06 


El contenido de la posición de memoria direccionada por el 
contenido del par de registros (HL) se rota a la izquierda un bit, 
y el resultado vuelve a almacenarse en esa posición. El conteni- 
do del bit 7 se lleva a la bandera de acarreo y al bit 0. 


Ж Тәні 
DATO | 


4 ciclos M; 15 estados T; 7.5 useg @ 2 MHz. 


Indirecto. 


s г H Фу NC 
өө jo] [ее 
С queda determinado por el bit 7 de la posición de memoria. 


RLC (HL) 


Antes: Después: 


371 


RLC (IX + d) 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 


372 


Rotación а la izquierda de la posición de memoria (IX + d) con 
copia al acarreo. 


РРЪРРР ер | byte 1: DD 


то [ото byte 2: CB 
[ИА byte 3: valor del desplazamiento 


ооо о [о [т 1 0 byte 4: 06 


El contenido de la posición de memoria direccionada por el 
contenido del registro IX más el valor del desplazamiento dado se 
rota a la izquierda, y el resultado vuelve a almacenarse en la mis- 
ma posición. El contenido del bit 7 se lleva a la bandera de aca- 
rreo y al bit 0. 


% 


\ 
\ 
_ 


My 
nu ND 
LA 
и | 
6 ciclos М; 23 estados T; 11.5 seg @ 2 MHz. 
Indexado. 
S Z H Qv N с 


OOMONODO 


C queda determinado por el bit 7 de la posición de memoria. 


Ejemplo: RLC (IX + 1) 
Antes: | Después: 
[4 E 72778 


CODIGO 
OBJETO 


373 


RLC (IY + d) 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 


374 


Rotación a la izquierda de la posición de memoria (ГУ + d) con 
copia al acarreo. 


пат то тр byte 1: FD 


НП] byte 2: СВ 
byte 3: valor del desplazamiento 
[о [о [о [о [о 1 [1 [о] byte 4: 06 


Е contenido de la posición de memoria direccionada рог el 
contenido del registro ГУ más el valor del desplazamiento dado se 
rota a la izquierda, y el resultado vuelve a almacenarse en la mis- 
ma posición. El contenido del bit 7 se lleva a la bandera de aca- 
rreo у al bit 0. 


Жи 
DATO | 


6 ciclos М; 23 estados Т; 11.5 useg @ 2 MHz. 
indexado. 


5 Фуу N с 


2 H 


C queda determinado por el bit 7 de:la posición de memoria. 


Ejemplo: ВГС (ТУ + 2) 


Antes: Después: 


PEN: РАГІ 


0021 
оға 8 
Са | озар “2 
Се = ЕТІ 
CODIGO 
OBJETO 


375 


RLD 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 


Direccionamiento: 


Banderas: 


376 


Rotación decimal a la izquierda. 


Los 4 bits inferiores de la posición de memoria direccionada 
por el contenido de HL se trasladan a los bits superiores de la 
misma posición. Los 4 bits superiores pasan a ocupar el lugar 
de los 4 inferiores del acumulador. Á su vez, éstos pasan a 
ocupar el lugar de los 4 bits inferiores de la posición de memo- 
ria especificada originalmente. Todos estos movimientos se eje- 
cutan simultáneamente. 


^ S 


EE 


5 ciclos M; 18 estados T; 9 useg @ 2 MHz. 


Indirecto. 


RLD 


Ejemplo: 


Después: 


Antes: 


B4F2 


B4F2 


са 
ва? в 7/7 


CODIGO 


OBJETO 


377 


RRs 


Rotación a la derecha de s a través del acarreo. 


Función: 
5 € 
Formato: г [1] 1[o[o[1[o[1[+] byte 1: CB 
ГГ ье 2 
(но [+[1[ofo[+[o[+[1] byte 1: CB 
(ооо 1|1|1]1|0] byte 2: 1E 
(IX + d) 1lol1l1l1lol1| byte 1: DD 
[1] +]ofo]rfo[1]1] byte 2: CB 
byte 3: valor del desplazamiento 
¡oJo]o]+[+[+[1]o] byte 4: 1E 
(IY +d) |! пато byte 1: FD 
1 ДЕЕ 1 ERE byte 2: CB 
P. ——————d-——— — —»| byte 3: valor del desplazamiento 
ооо ПРИ [o]. byte 4: 1E 
r puede ser: 
A — 111 E — 011 
B — 000 H — 100 
С — 001 L — 101 
D — 010 
Descripción: El contenido de la posición determinada por el operando espe- 


Flujo de datos: 


378 


cificado se desplaza a la derecha. El contenido de la bandera 
de acarreo pasa al bit 7, y el del bit O a la mencionada bandera. 
El resultado final vuelve a almacenarse en la posición de parti- 
da; s se define en la descripción de instrucciones RLC similares. 


Tiempo: 


Direccionamiento: 


Códigos byte: 
Banderas: 


Ejemplo: 


ME 
рац 


CODIGO 
OBJETO 


с изед 
5 Ciclos M Estados T @ 2 MHz 
r 2 8 4 
(HD) 4 15 7.5 
(IX + d) 6 23 | 
(IY + d) 6 23 11.5 


r: implícito; (HL): indirecto; (IX + d), (IY + d): indexado. 


RR r: 


B C 


г: A D E H L 
ca [re [19 ТА В ПС | 


[ө[ө[ Le] е: е 


С queda determinado por el bit 0 del dato fuente. 


RR H 
Antes: Después: 
Ae ][ ат | А КЁ 


379 


RRA 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


Tiempo: 
Direccionamiento: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


380 


Rotación a la derecha del acumulador a través del acarreo. 


CCE ae 


El contenido del acumulador se desplaza un bit hacia la dere- 
cha. El contenido de la bandera de acarreo pasa al bit 7, y el 
del bit O a la mencionada bandera (rotación de 9 bits). 


A 


| ciclo M; 4 estados T; 2 useg @ 2 MHz. 
Implícito. 


52 H PN с 
о о1Ф 
С queda determinado por el bit 0 de А. 


RRA 
Antes: Después: 
A F AEB TAE LÍO 


Nota: Esta instrucción es casi idéntica a RR A. Se incluye para 
garantizar la compatibilidad con el 8080. 


RRCs 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


(HL) 


(IX + d) 


(IY + d) 


Rotación a la derecha de s con copia al acarreo. 


5 с 


s puede ser: г, (HL), (IX + d), (IY + d). 


1 | o [1 ]o [1] byte 1: CB 


110 
Го То То 0 (ELI byte 2 


ооо [+ [1] byte 
0 ШЕ Tv [o byte 


ЕЕГ] bye 
[|| byte 
i UI byte 
Б ИШЕ 2 
ЖЛЕ БҮ ЕК byte 
A e] e 
as get] e 
HE 0 o | BERE [^] byte 
r puede ser: 
A — 111 
B — 000 
C — 001 
D — 010 


: CB 
: OE 
: DD 
: СВ 


: valor del desplazamiento 


ОЕ 


: FD 
: СВ 
: valor del desplazamiento 


: OE 


E — 011 
H — 100 
L — 101 


El contenido de la posición determinada por el operando espe- 
cificado se rota a la derecha, y el resultado vuelve a almacenar- 
se en la misma posición. El contenido del bit O pasa a la 
bandera de acarreo y al bit 7; s se define en la descripción de 


instrucciones RLC similares. 


I о Фр 
- мот 


381 


Tiempo: - _ E Ұлы 


іле useg 
5 Ciclos М Estados T o 2 MHz 
r 2 8 4 
(HL) 4 15 7.5 
(IX + d) 6 23 11.5 
(IY + d) 6 23 11.5 
Direccionamiento: r: implícito; (HL): indirecto; (IX + d), (IY + d): indexado. 
Códigos byte: RRC r 
TE АВС. DE н 1 
св оғ [ов [09 [ол Гов ос ор | 
Banderas: s z H (@ ум с 


e 0 0 e» e 
C queda determinado por el bit O del dato fuente. 
Ejemplo: RRC (HL) 


Antes: Después: 


Гв | АР" 
н 3FF2 L JE 3FF2 | 


3 ж бо 
FF2 7 
| oœ | —_ [жес | 


CODIGO 
OBJETO 


RRCA Rotación a la derecha del acumulador con copia al acarreo. 


Función: 


Formato: НОВО ЮО ОЕ 


Descripción: El contenido del acumulador se rota un bit hacia la derecha. El 
contenido del bit O pasa a la bandera de acarreo y al bit 7. 


Flujo de datos: 


Tiempo: 1 ciclo М; 4 estados T; 2 useg @ 2 MHz. 
Direccionamiento: Implicito. 
Banderas: 5 2 H PH N C 


|T tel | lele 


C queda determinado por el bit 0 de A. 


Ejemplo: RRCA 
Antes: Después: 
aos [ж] РРР 
CODIGO 
OBJETO 


RRD Rotación decimal a la derecha. 


Función: po 
ES рф Чы 


Formato: Saa] byte 1: ED 
оо јој [т] byte 2: 67 


Descripción: Los 4 bits superiores de la posición de memoria direccionada 
por el contenido del par de registro HL pasan a ocupar el lugar 
de los 4 inferiores de esa posición. Estos 4 bits inferiores pasan, 
a su vez, a ocupar el lugar de los 4 inferiores del acumulador. А 
su vez, estos 4 se trasladan al lugar de los 4 superiores de la 
posición de memoria especificada originalmente. Todos estos 
movimientos se ejecutan simultáneamente. 


Flujo de datos: 


А 

B 

D 

H 
Tiempo: 5 ciclos M; 18 estados T; 9 useg 2 MHz. 
Direccionamiento: Indirecto. 


Banderas: 


СГІЕСЕСІНЕ 


Ejemplo: RRD 


FEB] 


CODIGO 
OBJETO 


fEB! 


385 


RST p Reinicio en p. 


Función: (SP — 1) = РС. р; (SP — 2) < PC,,; SP < SP — 2; PC, + 0; 
РС», P 

Formato: Та pta 

Descripción: El contenido del contador del programa se empuja en la pila tal 


como se describe en las instrucciones РОЗН. El valor especifica- 
do de p se carga en el PC, y la siguiente instrucción se toma de 
esta nueva dirección; p puede ser: 


00H — 000 20H — 100 
08H — 001 28H — 101 
10H — 010 30H — 110 
18H — 011 38H — 111 


La instrucción ejecuta un salto a cualquiera de las ocho direc- 


ciones de inicio de la memoria inferior, y sólo necesita un byte. 
Puede utilizarse como respuesta rápida a una interrupción. 


Flujo de datos: A 
B с 
D E 
H L 


PIDA 


е? ты г 


Tiempo: 3 ciclos М; 11 estados T; 5.5 useg @ 2 MHz. 
Direccionamiento: Indirecto. 


Códigos byte: 


p: 00 08 10 18 20 28 30 38 


с? | се | о7 | oF | ЕЙ | eF I | | 


Banderas: CTI] 7 ІЗІГІ 1 (efecto nulo) 


Ejemplo: RST 38H 
Antes: Después: 
ках] «РЕ 
sP ғ? % 


0269 51 0269 mo 
CUN 
CODIGO 0268 03 0268 


== 9 
| 


ОВЈЕТО 


SBC А, $ Resta con acarreo del acumulador y el operando especificado, 


Función: А-А-5- С 
Formato: $ puede ser: г, n, (HL), (IX + d) o (IY + d) 
r filoioli FETES 
n byte 1: DE 
cap E: -| byte 2: dato inmediato 


(HL) [']ejo]r[r]r[r]o). byte 1: 9E 
asocia) ye da DD 


{о [о тъттор 9E 
— 4—— ———.| byte 3: valor del desplazamiento 
ay +d) РРРРРР ОТ | bye 1: FD 
11050|1|[1]|1]1]0| byte 2: 9E 
- d — | byte 3: valor del desplazamiento 


r puede ser: 


А — 111 Е — 011 
B — 000 H — 100 
C — 001 L — 101 
D — 010 
Descripción: El operando especificado s, al que se suma el contenido de la 


bandera de acarreo, se resta del contenido del acumulador y el 
resultado se almacena en el acumulador; s se define en la 
descripción de instrucciones ADD similares. 


Flujo de datos: 


Г 
m 
ed 
Кеа 


А 


388 


Tiempo: 


TT изед 

5 Ciclos M Estados T a 2 MHz 

r 1 4 2 

n 2 7 3.5 

(HL) 2 7 3.5 

(IX + а) 5 19 9.5 

(IY + d) 5 19 9.5 
Direccionamiento: r: implícito; n: inmediato; (HL): indirecto; (IX + d), (IY + d): 

indexado. 

Códigos byte: SBC A,r 


Banderas: s z H PAN с 
Ejemplo: SBC A,(HL) 
Antes: Después: 
А в [ s p А А 
2600 | SF | "m 
ET == а 
CODIGO 
OBJETO 


389 


SBC HL, $$ Resta con acarreo de HL y el par de registros ss. 


Función: HL - HL — $5 – С 


Formato: D]el | То +] byte 1: ED 
БТР) вие 2 

Descripción: El contenido del par de registros especificado, al que se suma el 
de la bandera de acarreo, se resta del contenido del par de 


registros HL, y el resultado se almacena de nuevo en HL; ss 
puede ser: 


BC — 00 HL — 10 
DE — 01 SP — 11 
Flujo de datos: 
йы y fe 
м КИЕ E 


= 


| Jes 


С 


| ud 


Tiempo: 4 ciclos M; 15 estados T; 7.5 useg @ 2 MHz. 
Direccionamiento: Implícito. 
Códigos byte: SS; BC DE HL SP 
ІІІ 
Banderas: 52 H PQ м с 


eje; Пее 


H se activa a 1 si hay acarreo del bit 12. 
C se activa a 1 si hay acarreo. 


Ejemplo: SBC HL, DE 
Antes: Después | 
E Са 1 ПРЕ: 
H| 34 1 ГЛ 
CODIGO 
OBJETO 


390 


SCF Pone a 1 la bandera de acarreo. 


Función: Cel 

Formato: [ o] o] я | (Те | 11 ІШ 37 

Descripción: Se pone a 1 la bandera de acarreo. 
Tiempo: 1 ciclo M; 4 estados T; 2 useg @ 2 MHz. 
Direccionamiento: Implicito. M 

Banderas: E PN C 


2 H 


SET b,s Activa a 1 el bit b del operando s. 


Función: S = 1 
Formato: 5: 
г 111|19|0|1|911)1 byte 1: СВ 
ales byte 2 
(HL) 1 тъ тото | то | 1 11 byte 1: СВ 
na pe RE aa] byte 2 
(IX + d) поет 11 јот byte 1: DD 
{тоот [о | 1 | 1 byte 2: СВ 


byte 3: valor del desplazamiento 
E E --гь-- sn [^] byte 4 
(IY + d) парта то | т byte 1: ЕО 
1 1 0919 1 9 1 byte 2 СВ 


1 

2 

5------4------ byte 3: valor del desplazamiento 
BE --ь-- 1 E [o byte 4: 


r puede ser: 
А — 111 E — 011 
B — 000 H — 100 
C — 001 L — 101 
D — 010 
b puede ser: 
0 — 000 4 — 100 
1 — 001 5 — 101 
2 — 010 6 — 110 
3 — 011 7111 
Descripción: El bit especificado de la posición determinada por s se activa a 


1; s se define en la descripción de instrucciones ВІТ similares. 


Flujo de datos: 


г о о >» 
о 


392 


Tiempo: 


А изед 
5 Ciclos M Estados T @ 2 MHz 
r 2 8 4 
(HL) 4 15 7.5 
(IX + d) 6 23 11.5 
(IY + d) 6 23 11.5 
Direccionamiento: г: implícito; (HL): indirecto; (IX + d), (IY + d): indexado. 
Códigos byte: SET b,r 
CB- e c5 
C9 [ca | [ca св EH 
DI ШЕ оз | Da | 
ОА | DB ша 
El | Е2 | ЕЗ ЕБ 
BE ІЗЕ 
| F2 | F3 | Fa | Fs | 
020030120 
SET b, (НО 
b: o 1 2 3 4 5 6 7 
SET b,(IX + d) | cs] сє | пе] гє | es | e | ve [re | | 
SET Ь, ПУ + d) 
| 
Banderas: 5 2 н PNN C 
(efecto nulo) 


Ejemplo: SET 7,A 
| Antes: 
| ( ге 


CODIGO 
OBJETO 


Después: 


222 


393 


SLA s 


- Función: 


Formato: 


Descripción: 


Flujo de datos: 


394 


(IY + d) 


Desplazamiento aritmético a la izquierda del operando s. 


CL 0 
с 5 
5: 
1000000008 
ппйпп = bre 
0000000008 
0900000002 
ШЕГЕ ШЕ byte t: 
E [+]o[o[+Te [1] byte 2: 
[esu ea 12 
СТТ] byte 4: 
ПОЮ 1 ofi] byte 1: 
LEIL Pe D] вие 2: 
ез a l bes 
0 то [о г | т 10 Буе 4 
г puede ser: 

A — 111 

B — 000 

C — 001 

D — 010 


: valor del desplazamiento 


26 
FD 
CB 


: valor del desplazamiento 
: 26 


E — 011 
H — 100 
L — 101 


El contenido de la posición de memoria determinada por el 
operando especificado se desplaza aritméticamente a la izquier- 
da, pasando el contenido del bit 7 a la bandera de acarreo e 
introduciendo un 0 en el bit 0. El resultado final vuelve a 
almacenarse en la posición original; s se define en la descripción 
de instrucciones RLC similares. 


тосе > 


= мот 


Tiempo: 


| e | useg 

5 Ciclos М Estados T @ 2 MHz 

r 2 8 4 

(HL) 4 15 7.5 

(IX + d) 6 23 11.5 

(IY + d) 6 23 11.5 
Direccionamiento: г: implícito; (HL): indirecto; (IX + d), (IY + d): indexado. 
Códigos byte: SLA r 


г A B C D E ПН 


L 
CB. 27 | 20| 21 | 22 | 23 24 |25 | 


Banderas: 5 Z H QvN с 
(ele! |:| е: е 
C queda determinado por el bit 7 del dato fuente. 
Ejemplo: SLA (HL) 
Antes: Después: 
[ETA | Ы 
н| OFF2 E Al оғ? |: 
OFF2 оғғ? 
E) 
CODIGO 
OBJETO 


395 


SRA s Desplazamiento aritmético a la derecha del operando s. 


Función: и Г] 
5 E 
Formato: 5 
r [+[o[o]+]o]+]> byte 1: CB 
Се [е ЕГ ье 2 
(HL) Paja alo TORNE byte 1: CB 
Б] byte 2: 2E 
ax+d ППС Г byte 1: DD 
EXE То [о |. | | | byte 2: СВ 
----4---- byte 3: valor del desplazamiento 
oo] [р] byte 4: 2E 
иү+а СЕСЕ byte 1: Ер 
ШЕ оо | 1 [o [+ [1] byte 2: CB 
-- d ---| byte 3: valor del desplazamiento 
ojo ШЕЗЕЛЕЕШЕН byte 4: 2Е 
r puede ser: 
A — 111 E — 011 
B — 000 H — 100 
C — 001 L — 101 
D — 010 
Descripción: El contenido de la posición determinada por el operando espe- 


cificado se desplaza aritméticamente a la derecha. El contenido 
del bit O se traslada a la bandera de acarreo y el del bit 7 
permanece invariable. El resultado fina] se almacena en la posi- 
ción original; s se define en la descripción de instrucciones RLC 


similares. 


Flujo de datos: 


396 


Tiempo: 


2 seg 

5 Ciclos М Estados Т @ 2 MHz 

r 2 8 4 

(HL) 4 15 7.5 

(IX + d) 6 23 11.5 

(IY d) 6 23 11.5 
Direccionamiento: г: implícito; (HL): indirecto; (IX + d), (IY + d): indexado. 
Códigos byte: SRA r 


ГА В C D E H L 
св. [27 | 28 | 29 24 | 28 |с 2D 


Banderas: s z H Фу м с 
[ele] T°] lelojej 
C queda determinado por el bit O del dato fuente. 
Ejemplo: SRA A 
Antes: Después: 
А æ | м | жуы 
>= 
CODIGO 
OBJETO 


397 


SRLs 


Función: 


Formato: 


Descripción: 


Flujo de datos: 


398 


(HL) 


(IX + d) 


(IY + d) 


Desplazamiento lógico a la derecha de s. 


0 7 ——m-0 Г] 
5 E 

8; 

EXE 0 ol! ШЕ Буе 1: СВ 

[olo DOR byte 2 

Е | ШЕ GERE byte 1: CB 

[о] о SEE ЛИП byte 2: 3E 

Е р] byte 1: DD 

Е ШЕ Но | | byte 2: CB 

[————4————]| byte 3: valor del desplazamiento 
о|0 ата 151 byte 4: ЗЕ 

РГ bre 1: £p 

E 112211121211) byte 2: СВ 

————4————] byte 3: valor del desplazamiento 
о оттто [о | byte 4: ЗЕ 


г puede ser: 


A — 111 E — 011 
B — 000 H - 100 
C — 001 L — 101 
D — 010 


El contenido de la posición determinada por el operando espe- 
cificado se desplaza lógicamente a la derecha. Se introduce un 
cero en el bit 7, y el contenido del bit 0 pasa a la bandera de 
acarreo. El resultado final se almacena en la posición original. 


> 


TOO 
пот 


Tiempo: 


5 Ciclos М Estados T @ a ME 

r 2 8 4 

(HL) 4 15 7.5 

(IX + d) 6 23 11.5 

(IY + d) 6 23 11.5 
Direccionamiento: r: implicito; (HL): indirecto; (IX + d), (IY + d): indexado. 
Códigos byte: SRL r 


r A B С D E Н L 


CB | 3F зв] 39 ЗА] эв] эс] 30] 


Banderas: " н Фум с 


ГІСІШШЕСІЕГІ 


С queda determinado por el bit O del dato fuente. 


Ejemplo: SRL E 
Antes: Después: 
c 8. ПЕР 
— Го 16 ГАУ 
-- 
CODIGO 
OBJETO 


399 


ЗОВ $ 


Función: 
Formato: 
r 
n 
(HL) 
(IX + d) 
(IY + d) 
Descripción: 


Flujo de datos: 


| 


Resta del operando 5 dél acumulador. 


А-А-$ 
s puede ser: г, n, (HL), (IX + d) o (IY + d) 
POE 


byte 1: D6 

byte 2: dato inmediato 
96 

byte 1: DD 


byte 2: 96 


byte 3: valor del desplazamiento 
byte 1: FD 


byte 2: 96 


byte 3: valor del desplazamiento 


еар а 
ІП Е Еее Е В 
* lb ELE E i 
Жү е ж ЕК ЕЕ ЕН 
НИЕДЕ Е 
НЕР ena MES 
a la AA E 
Le] LJ LU lol E [e] Le] 


А — 111 E — 011 
B — 000 Н — 100 
C — 001 L — 101 
D — 010 


El operando especificado s se resta del acumulador, y el resulta- 
do se almacena en éste. El operando s se define en la descrip- 
ción de instrucciones ADD similares. 


н 
| 
| 
| 


Tiempo: 


Direccionamiento: 


Códigos byte: 


Banderas: 


Ejemplo: 


CODIGO 
OBJETO 


5 Ciclos М Estados T a Е 
r 1 4 2 

n 2 7 3.5 
(НГ) 2 7 3.5 

(IX + d) 5 19 9.5 

(IX + d) 5 19 9.5 


г: implícito; n: inmediato; (HL): indirecto; (IX + d), (IY + d): 
indexado. 


5 Z H PON С 

е jej jej: [ө 

SUB B 

Antes: Después: 
АГ 3 | АЙ 
== "mcum 


XOR s 


Función: 


Formato: 


Descripción: 


Flujo de datos: 
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(IY + d) 


PERLI [AS bye ЕЕ 
byte 2: dato inmediato 

EE TE DR 

Торо [ie] byte 2: AE 

byte 3: valor del desplazamiento 
РОО byte 1: FD 
EEBUDEHEMN 2 AB 

byte 3: valor del desplazamiento 


r puede ser 
А — 111 E — 011 
В — 000. H — 100 
C — 001 L — 101 
D — 010 


El acumulador y el operando especificado s se someten a la 
operación “O” exclusiva, y el resultado se almacena en el acu- 
mulador; s se define en la descripción de instrucciones ADD 


Tiempo: 


Direccionamiento: 


Códigos byte: 


Banderas: 
Ejemplo: 


re 
EN | 
дете 
| 


CODIGO 
OBJETO 


S Ciclos M 
r 1 
n 2 
(HL) 2 
(IX + d) 5 
(IY 4 d) 5 


г: implícito; n: inmediato; (HL): indirecto; (IX + d). (ТҮ + d): 
indexado. 


XOR r 


r: A B C D E HL 


or pepe Tas epo 


S Z н  Á (VN с 


ее [o] өсе 


XOR B1H 
Antes: Después: 
АГ s | АЙ 
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Ау 
A 


LOTITI 


жән 
E odi 


HARE 


fr 

е 
ES 
зе 
E d 
E 
E 
LJ 
жш 
же 
ж 
= 


de 
ше 
bud 
ве 
ak. 
ж 
ж 
#6 
Ld 
Ld 
qu 
ж 
җе 


Técnicas de 


direccionamiento 


Introducción 


Veremos en este capítulo la teoría general del direcciona- 
miento y las diversas técnicas desarrolladas para facilitar la 
recuperación de datos. Pasaremos a continuación a examinar 
los mecanismos de direccionamiento especificos del Z80, cen- 
trándonos en particular en sus ventajas y sus inconvenientes. En 
la última parte —dedicada a las aplicaciones— se familiarizará 
al lector con los posibles intercambios entre diferentes técnicas 
de direccionamiento. 

Como el Z30 tiene varios registros de 16 bits, además del 
contador del programa, que sirven para especificar direcciones, 
es importante que el usuario del mismo conozca los diversos 
modos de direccionamiento y, en particular, el uso del registro 
de índice. En una primera fase puede prescindirse de los proce- 
dimientos más complejos, pero hay que tener en cuenta que 
para desarrollar programas para este microprocesador son úti- 
les todas las formas de direccionamiento. Pasemos ya a estudiar 
las opciones disponibles. 
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Modos de direccionamiento 


Se llama direccionamiento a la especificación, dentro de una 
instrucción, de la posición del operando sobre el que actuará la 
misma. Los modos de direccionamiento que vamos a estudiar 
ahora se ilustran en la figura 5.1. 


DIRECCIONAMIENTO IMPLICITO (O “POR REGISTRO”) 


Las instrucciones que operan exclusivamente con registros 
suelen utilizar el direccionamiento implícito, como ilustra la figu- 
ra 5.1. El nombre se debe a que la instrucción no contiene de 
forma explicita la dirección del operando, sino que su código de 
operación especifica uno o más registros, por lo general el 
acumulador y algún otro u otros. Dado que no hay muchos 
registros internos (habitualmente ocho), esta técnica exige pocos 
bits; de hecho, bastan tres bits dentro de la instrucción para 
señalar a uno de los ocho registros internos; por tanto, tales 
instrucciones pueden, por lo general, codificarse totalmente con 
ocho bits. Esto constituye una ventaja importante, porque una 
instrucción de un byte se ejecuta casi siempre más rápidamente 
que otra de dos o tres. 

Veamos un ejemplo de instrucción implícita: 


LD А, B 


que especifica “transferir el contenido de B a A” (carga de A a 
partir de B). 


DIRECCIONAMIENTO INMEDIATO 


También se muestra en la figura 5.1. Al código de operación 
de ocho bits sigue un literal (una constante) de 8 ó 16 bits. Este 
tipo de instrucción es necesario para, por ejemplo, cargar un 
valor de 8 bits en un registro de 8 bits. Como el microprocesa- 
dor dispone de registros de 16 bits, será también necesario 
cargar literales de esa longitud. He aquí una instrucción inme- 
diata: : 


ADD A,0H 


» 
La segunda palabra de la instrucción contiene el literal “0”, 
que se suma al acumulador. 


7 0 
IMPLICITO CODIGO OPA IR 


CODIGO OP 


1 
| LITERAL | 
J 


INMEDIATO 


AMPLIADO/ABSOLUTO CODIGO OP 


DIRECCION DE 


16 BITS 


DIRECTO/ABREVIADO 


CODIGO OP 
DIRECCION CORTA 


1 
| CODIGO OP ! 


И 

CODIGO OP |ВЕСХ 
DESPLAZAMIENTO 

Figura 5.1 

Modos de direccionamiento 


[ t 
fundamentales. Cona a J 


INDEXADO 


DIRECCIONAMIENTO ABSOLUTO 


Por lo general, hace referencia a la recuperación o al alma- 
cenamiento de datos de o en la memoria. Al código de opera- 
ción sigue una dirección de 16 bits, de manera que esta forma 
de direccionamiento obliga a trabajar con instrucciones de tres 
bytes. Veamos un ejemplo: 


LD (1234H), A 
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Especifica que debe almacenarse el contenido del acumulador 
en la posición hexadecimal de memoria “1234”. 

El inconveniente de esta forma de direccionamiento es la 
mencionada obligación de trabajar con instrucciones de tres 
bytes. Para mejorar la eficacia del microprocesador, se cuenta, a 
veces, con otro modo de direccionamiento llamado direcciona- 
miento directo, que sólo necesita una palabra. 


DIRECCIONAMIENTO DIRECTO (O “ABREVIADO”) 


En este modo, al código de operación sigue una dirección de 
8 bits (también aparece en la figura 5.1). Tiene la ventaja de que 
equivale al direccionamiento absoluto, pero sólo con dos bytes 
en lugar de tres. El inconveniente es que está limitado a las 
direcciones comprendidas entre O y 255 o entre — 128 y + 127. 
En el primer caso (“página cero”), se habla también de direccio- 
namiento abreviado o direccionamiento de página 0. Cuando se 
dispone de esta posibilidad, al direccionamiento absoluto se le 
llama, por contraste, direccionamiento ampliado. El intervalo 
— 128 a + 127 se utiliza en las instrucciones de bifurcación, y 
se llama direccionamiento relativo. 


DIRECCIONAMIENTO RELATIVO 


Las instrucciones de salto o bifurcación normal necesitan 8 
bits para el código de operación más 16 bits para la dirección a 
la que debe saltar el programa. Como en el ejemplo anterior, 
este modo tiene el inconveniente de que obliga a usar tres 
palabras; por tanto, tres ciclos de memoria. El direccionamiento 
relativo se conforma con dos, y garantiza una bifurcación más 
rápida. La primera palabra es la especificación de la posición de 
salto, comprendida, por lo general, en la verificación que se 
efectúa; la segunda es un desplazamiento, positivo o negativo, 
que permite a la instrucción avanzar hasta 127 posiciones (siete 
bits) o retroceder hasta 128 (por lo general, de + 129 a — 126, 
porque el PC se habrá incrementado en 2). Como casi todos los 
bucles son breves, la mayor parte de ellos pueden controlarse 
por direccionamiento relativo, lo que mejora considerablemente 
la velocidad y la eficacia de tales subrutinas. Como ejemplo, ya 
hemos utilizado la instrucción JR NC, que especifica “saltar si 
no hay acarreo” a una posición situada a no más de 127 bytes 
de la instrucción de bifurcación (más exactamente, comprendida 
entre + 129 y - 126). 


Figura 5.2 
Direccionamiento 
ción). 


(preindexa- 


Esta técnica tiene dos ventajas: mayor velocidad, debida al 
uso de menos bytes, y facilidad de redireccionamiento del pro- 
grama, que no depende de direcciones absolutas. 


DIRECCIONAMIENTO INDEXADO 


Se emplea este modo para acceder en secuencia a todos los 
elementos de un bloque o de una tabla; demostraremos la 
técnica hacia el final de este capítulo con ayuda de algunos 
ejemplos. La instrucción especifica una dirección y un registro 
de índice, cuyo contenido se suma a la dirección para obtener la 
dirección definitiva. De esta forma, la dirección puede ser el 
principio de una tabla situada en memoria, que se recorre en 
secuencia elemento tras elemento por medio del registro de 
índice (naturalmente, hacen falta instrucciones de incremento y 
decremento para dicho registro). En la práctica, el tamaño del 
registro de indice, el de la dirección o el del campo de desplaza- 
miento suelen tener un límite. 


PREINDEXACION Y POSTINDEXACION 


La preindexación es la forma de indexación normal; en ella, 
la dirección final es la suma de un desplazamiento o dirección y 
del registro de índice. Se muestra en la figura 5.2, con un campo 
de desplazamiento de 8 bits y un registro de indice de 16 bits. 


REGISTRO DE INDICE 


BASE 


desplazamiento 


MEMORIA 


La postindexación considera el contenido del campo de 
desplazamiento como la dirección del desplazamiento real en 
lugar de como el desplazamiento propiamente dicho, como 
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ilustra la figura 5.3. En esta modalidad, la dirección final es la 
suma del contenido del registro de índice y de la palabra de 
memoria designada por el campo de desplazamiento. Esta técnica 
es, en realidad, una combinación de direccionamiento indirecto, 
que estudiaremos a continuación, y preindexación. 


MEMORIA Y (índice) 


PUNTERO ; 


MEMORIA 


CODIGO OP 


DIRECCION 


DIRECCION 
FINAL 
DE 16 BITS 


PUNTERO - BASE 


Figura 5.3 
Direccionamiento indexado in- 
directo (postindexación). 


DIRECCIONAMIENTO INDIRECTO 


Ya hemos visto que a veces dos subrutinas tienen que inter- 
cambiar un gran volumen de datos almacenados en memoria. 
Todavía es más frecuente que varios programas, o varias subru- 
tinas, necesiten acceso a un bloque de información comün. Para 
proteger el cuerpo del programa es aconsejable que dicho blo- 
que no se mantenga en una posición fija de memoria; en efecto, 
el tamafio del mismo puede aumentar o disminuir a lo largo del 
tiempo, por lo que debe residir en diversas zonas de la memo- 
ria, según su volumen. En estas circunstancias no sería práctico 
acceder al bloque mediante direccionamiento absoluto, porque 
habría que escribir el programa completo cada vez que cambia- 
se de posición. 

La solución está en situar la dirección de partida del bloque 
en una posición fija de memoria. Es una solución similar a la 
adoptada cuando varias personas necesitan tener acceso aeuna 
casa de la que sólo hay una llave: esconder ésta debajo del 
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Figura 5.4 
Direccionamiento indirecto. 


felpudo. Todos los usuarios saben lo que tienen que hacer para 
dar con la llave (mirar bajo el felpudo) o con la dirección en la 
que se celebrará una reunión (este caso está más próximo al 
que nos ocupa). Por tanto, el direccionamiento indirecto utiliza 
habitualmente un código de operación al que sigue una direc- 
ción de 16 bits, que se utiliza para recuperar una palabra de la 
memoria. Será casi siempre una palabra de 16 bits (en nuestro 
caso, dos bytes), puesto que se trata de una dirección; la situa- 
ción se ilustra en la figura 5.4: los dos bytes de la dirección 
especificada A1 contienen "A2", que se interpreta como la ver- 
dadera dirección del dato al que se desea acceder. 


INSTRUCCION MEMORIA 
CODIGO OP 


DIRECCION DIRECCION 


INDIRECTA A FINAL (A,) 


El direccionamiento indirecto es particularmente ütil en to- 
das las situaciones en que se emplean punteros, porque así 
las diversas áreas del programa pueden dirigirse a dichos pun- 
teros para acceder a una palabra o a un bloque de datos con 
comodidad y elegancia. La dirección final también puede obte- 
nerse sefialando dentro de la instrucción un registro de 16 bits 
en el que esté contenida; esto se llama "registro indirecto". 


COMBINACIONES DE MODOS 


Las técnicas de direccionamiento que hemos visto pueden 
combinarse. En un esquema de direccionamiento completamen- 
te general debe ser posible trabajar en varios niveles; asi, la 
dirección A2 podría interpretarse, a su vez, como una dirección 
indirecta, etc. 
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El direccionamiento indexado puede combinarse con el acce- 
so indirecto, lo que permite acceder rápidamente a la palabra n 
de un bloque de datos si se sabe dónde está el puntero que 
señala la dirección de partida (véase figura 5.2). 

Una vez familiarizados con los modos de direccionamiento 
habituales, estudiaremos los del Z80, que ofrece bastantes posi- 
bilidades. Hay que tener en cuenta que los microprocesadores, 
debido a la limitación de la complejidad del uP, que ha de 
estar contenida en una sola pastilla, suelen ofrecer tan sólo un 
pequefio subconjunto de las técnicas estudiadas. 


Modos de direccionamiento del Z80 
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DIRECCIONAMIENTO IMPLICITO (Z80) 


Sobre todo, se emplea en instrucciones de un byte que 
operan con registros internos. Estas instrucciones se ejecutan en 
un solo ciclo de máquina. 

Utilizan direccionamiento implícito (o “рог registro"): 
LDr,r;ADDA,r; ADCAS; SUB s; SBC As; AND s; OR s; 
XOR s; CP s; INC r. 

Zilog diferencia entre “direccionamiento por registro” y “di- 
reccionamiento implicito”. Según este enfoque, el direcciona- 
miento implícito se limita a las instrucciones que no tienen un 
campo especifico para señalar a un registro interno, lo que 
introduce, en realidad, un nuevo modo de direccionamiento. 
Estos, de hecho, no bastan para definir las posibilidades de un 
microprocesador. 


DIRECCIONAMIENTO INMEDIATO (2780) 


Como el Z80 dispone de registros de longitud simple (8 bits) 
y de pares de registros de longitud doble (16 bits), ofrece dos 
tipos de direccionamiento inmediato con literales de 8 o de 16 
bits e instrucciones de dos o tres bytes. El segundo byte (y a 
veces, el tercero) contiene el código de operación seguido de la 
constante o literal que ha de cargarse en un registro o utilizarse 
en una operación. Son excepciones LD IX y LD IY, que necesi- 
tan códigos de operación de 16 bits. 


Son ejemplos de instrucciones que utilizan direccionamiento 
. Inmediato: 


LD R, n (dos bytes). 
LD dd,nn (tres bytes). 


ADD A,n (dos bytes). 


Cuando el literal tiene dos bytes, el modo se llama en el Z80 
”inmediato ampliado”. 


DIRECCIONAMIENTO ABSOLUTO O “AMPLIADO” 
(280) 


El direccionamiento absoluto precisa, por definición, tres 
bytes; el primero es el código de operación, y los dos siguientes, 
la dirección de 16 bits que especifica la posición de memoria (la 
“dirección absoluta”). 

Por contraste con el “direccionamiento directo” (dirección 
de 8 bits), este modo se llama también “direccionamiento am- 
pliado”. 

Utilizan, entre otras, direccionamiento ampliado las instruc- 
ciones 


LD НЕ, (пп) y JP nn 


donde nn representa la dirección de memoria de 16 bits y (nn) 
el contenido de la posición especificada. 


DIRECCIONAMIENTO DE PAGINA CERO 
MODIFICADO (780) 


El Z80 sólo dispone de direccionamiento de página cero 
para la instrucción RST, y se llama «direccionamiento de pági- 
na cero modificado”. 

Dicha instrucción contiene un campo de 3 bits en las posi- 
ciones 5, 4 y 3 que señala una de las ocho posiciones de la 
memoria de página 0. La dirección real es bsb,b3000, y se carga 
en el PC. Como sólo ocupa un byte, la instrucción se ejecuta 
rápidamente, y se genera fácilmente en el soporte físico. Suele 
emplearse para responder a interrupciones múltiples (hasta 
ocho). Tiene el inconveniente de que, o bien se limita la secuen- 
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cia de ejecución a ocho posiciones, o bien se combina con un 
salto que elimina la ventaja de la velocidad, porque las ocho 
direcciones de bifurcación están separadas por una distancia de 
8 bytes. 


DIRECCIONAMIENTO RELATIVO (780) 


Por definición, el direccionamiento relativo exige dos bytes; 
el primero es el código de operación de “salto relativo”, el 
segundo especifica el desplazamiento y su signo. 

Esta instrucción se codifica “JR” para diferenciarla de la de 
salto absoluto. 

Desde el punto de vista del tiempo de ejecución, la instruc- 
ción debe considerarse con cuidado. Si el resultado de una 
verificación es negativo, es decir, si no hay salto, consume sólo 
siete ciclos T, porque el contador del programa está ya señalan- 
do la instrucción siguiente. 

Pero si el resultado de la comprobación es afirmativo, es 
decir, si hay salto, la instrucción necesita 12 estados T, porque 
hay que calcular una nueva dirección y cargarla en el contador 
del programa. 

Para calcular lo que tarda en ejecutarse un segmento de 
programa, hay que andarse con cuidado. Cuando no se tiene la 
seguridad de si un salto va o no a efectuarse, téngase en cuenta 
que la instrucción necesitará en unos casos 12 estados T (la 
condición se satisface) y en otros sólo 7 (la condición no se 
satisface). 

Por tanto, al escribir un bucle se conseguirá mayor rapidez 
de ejecución con una instrucción JR (salto relativo) si la condi- 
ción que ha de verificarse no suele cumplirse (por ejemplo, la 
condición de que el contador no sea 0). 

Si JR's se emplea fuera de un bucle y la condición que se 
verifica es desconocida, la duración suele calcularse a partir de 
un tiempo medio. 

Este problema del tiempo no ocurre en el salto incondicio- 
nal JR e, que no verifica ninguna condición y dura siempre 12 
estados T. 


DIRECCIONAMIENTO INDEXADO (780) 


Este modo de direccionamiento no existía en el 8080, y 
constituye una novedad del Z80 (al igual que los dos registros 
de índice). En consecuencia, es necesario añadir un byte extra al 
código de operación de este microprocesador, que pasa a tener 


Figura 5.5 

Direccionamiento indexado con 
código de operación de 2 by- 
tes. 


16 bits (LDIR es otro ejemplo de código de operación de 16 
bits). En la figura 5.5 se muestra la estructura de una instruc- 
ción indexada. 


CODIGO OP BYTE 1 
BYTE 2 
BYTE 3 
І 
1 LITERAL 1 ВҮТЕ4 
К...---...-.-....-. E 


Son instrucciones compatibles con el direccionamiento inde- 
xado 


LD, ADD, INC, RLC, BIT, SET, CP y algunas otras. 


Es una técnica que se utiliia mucho en programas que 
manipulan constantemente bloques de datos, tablas o listas. 


DIRECCIONAMIENTO INDIRECTO (280) 


El Z80 tiene cierta capacidad de direccionamiento indirecto, 
que se llama “direccionamiento indirecto por registro”. En este 
modo, los pares de registros de 16 bits BC, DE y HL pueden 
emplearse como direcciones de memoria. 

Cuando sefialan un dato de 16 bits, lo hacen siempre a su 
mitad inferior; la superior reside en la dirección siguiente. 


COMBINACIONES DE MODOS 


Básicamente no existen, aunque las instrucciones que se 
refieren a dos operandos pueden emplear una forma de direc- 
cionamiento diferente para cada uno de ellos. 

Así, una instrucción de carga o aritmética puede acceder a 
un operando en modo inmediato y a otro por medio de un 
índice. 

Como veremos en el próximo apartado, el mecanismo de 
direccionamiento de bit puede acceder a un byte de 8 bits 
mediante tres modos de direccionamiento. Los mecanismos 
compatibles con las diferentes instrucciones se incluyeron en las 
descripciones del capítulo anterior. 


ENT: 


Empleo de los 
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DIRECCIONAMIENTO DE BIT 


Dado que por direccionamiento se entiende el acceso a un 
byte, este mecanismo no se considera como tal, aunque, en 
cualquier caso, se trata de un recurso valioso. En la nomencla- 
tura de Zilog sí que se describe como un “modo de direcciona- 
miento”, y aquí nos atendremos a ese punto de vista. Es una 
característica específica del Z80 que no existe en el 8080. 

E] direccionamiento de bit es un mecanismo de acceso a bits 
específicos. El Z80 dispone de instrucciones especiales para acti- 
var, desactivar y verificar bits específicos de un registro o una 
posición de memoria. El byte afectado admite el acceso por tres 
modos de direccionamiento: registro, registro indirecto e inde- 
xado. Dentro del código de operación se usan tres bits para 
seleccionar uno de los ocho bits. 


modos de direccionamiento del Z80 


DIRECCIONAMIENTO LARGO Y ABREVIADO 


Ya hemos utilizado instrucciones de salto relativo en varios 
programas, instrucciones que se explican por sí mismas. Pero se 
plantea una pregunta interesante: ¿qué podemos hacer si el 


intervalo permisible de bifurcación no es suriciente para nues- 


tras necesidades? En muchos microprocesadores la solución está 
en usar el llamado salto largo, que no es sino el salto a una 
posición de memoria que contiene una especificación de salto 
absoluta. o "larga": 


JR NC,$ + 3 BIFURCACION A LA DIRECCION 
EN CURSO + 3, SI С ES CERO 

JP LEJOS SALTO A LEJOS, EN CASO CON- 
TRARIO 


(INSTRUCCION SIGUIENTE) 


El anterior programa de dos líneas provoca la bifurcación a 
la posición LEJOS siempre que el acarreo no sea nulo. En el 
caso del Z80 puede usarse JP en lugar de JR para verificar 
todas las condiciones y solucionar este problema. 


Figura 5.6 
Diagrama de flujo de búsqueda 
de carácter. 


USO DE LA INDEXACION PARA ACCEDER A 
BLOQUES SECUENCIALES 


La indexación se usa, sobre todo, para direccionar las posi- 
ciones sucesivas de una tabla, con la sola limitación de contar 
con una longitud máxima inferior a 256 para que el desplaza- 
miento pueda residir en un registro de índice de 8 bits. 

Ya hemos aprendido a buscar un carácter. Ahora investiga- 
remos una tabla de 100 elementos para averiguar si contiene el 
elemento “*”, La dirección de partida de dicha tabla, que sólo 
tiene 100 elementos, se llama BASE. El programa aparece aqui 
debajo, y el diagrama de flujo correspondiente, en la figura 5.6. 


BUSCAR LD IX, BASE 
LD А,” 
LD В,СОЕМТА 
PRUEBA CP (IX) 
JR Z,LOHALLE 
INC IX 
DEC B 


JR NZ,PRUEBA 
LOHALLE ks 


INICIALIZAR 
A ELEMENTO 0 
LEER SIGUIENTE 

ELEMENTO 


ASTERISCO HALLADO 


APUNTAR AL 
SIGUIENTE ELEMENTO 


NO HALLADO 


En la sección de transferencia de bloques veremos una ver- 
sión mejorada con la instrucción DJNZ. 
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RUTINA DE TRANSFERENCIA DE BLOQUES PARA 
MENOS DE 256 ELEMENTOS 


Llamaremos CUENTA al número de elementos del bloque 
que deseamos transferir, número que se supone inferior a 256. 
DESDE es la dirección base del bloque. HASTA es la base del 
área de memoria a la que debe llevarse. El algoritmo es muy 
sencillo: se trata de mover las palabras una por una y de 
mantener un registro de la que estamos moviendo, almacenan- 
do su posición en el contador C. El programa es éste: 


MOVBLQ LD IX, DESDE 
LD IY, HASTA 
LD C, CUENTA 


BUCLE LD  A,(IX) TOMAR PALA- 
BRA 
LD  (IY.A 
INC IX 
INC IY 
DEC C 


JR NZ, BUCLE 
Empecemos a analizarlo por la primera parte: 


MOVBLQ LD IX, DESDE 
LD IY, HASTA 
LD С, СОЕМТА 


Estas tres instrucciones inicializan los registros ІХ, ТҮ у С, 
respectivamente, como ilustra la figura 5.7. El registro de índice 
IX se usa como apuntador de fuente y se incrementa regular- 
mente. El ГУ es el apuntador de destino, y también se incremen- 
ta con regularidad. En el registro C se carga el número de 
elementos que va a transferirse (limitado a un máximo de 256, 
porque se trata de un registro de 8 bits) y se decrementa 
regularmente. Cuando C alcance el valor 0, todos los elementos 
habrán sido transferidos. Las dos instrucciones que vienen a 
continuación 


BUCLE LD А.(ІХ) 
LD (ГА 


cargan el contenido de la posición de memoria señalada por IX 
en el acumulador, y a continuación lo transfieren a la posición 
de memoria indicada por el registro IY; en otras palabras, 


Figura 5.7 


Transferencia de bloques: ini- 


cialización del registro. 


estas dos instrucciones transfieren un elemento del bloque 
fuente al bloque destino. A continuación se incrementan los dos 


registros de índice: 


INC IX 
INC IY 


y se decrementa el registro contador: 


DEC C 


Por último, si el contador no es 0, se vuelve a empezar en bucle: 


JR NZ, BUCLE 


EXE 


ши 
” 


OL SE 
— ] 


en d 


El programa es un ejemplo de una posible aplicación de los 
registros de índice. Vamos a compararlo ahora con otro igual 
escrito para el microprocesador 6502 de tecnología MOS, que 
también dispone de capacidad de indexación, pero que sigue 
convenciones diferentes (es decir, que tiene límites de indexación 
diferentes). El programa en cuestión es el siguiente: 


LDX 
BUCLE LDA 
STA 
DEX 
BNE 


H NUMERO 
DESDE, X 
HASTA. X 
BUCLE 
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Sin necesidad de entrar en detalles, salta a la vista que es 
mucho más corto que el anterior. La razón está en que el 
registro de indice X se utiliza como desplazamiento variable, 
mientras que DESDE y HASTA son direcciones fuente y desti- 
no fijas. 

El ejemplo revela que, aunque la indexación sea un recurso 
poderoso, no da lugar necesariamente a programas eficaces, 
debido a las limitaciones de direccionamiento que imponen 
ciertos microprocesadores. Una genuina indexación de tipo ge- 
neral exige disponer de un desplazamiento o dirección de 16 
bits y de un registro de índice de la misma longitud. 

No obstante, hay que insistir en que el Z80 resuelve el 
problema mediante instrucciones especializadas. Describiremos 
a continuación una transferencia de bloques de tipo general que 
se ejecuta con sólo cuatro instrucciones. Pero sugerimos al 
lector, antes de pasar a analizarla, que resuelva los siguientes 
ejercicios, que le familiarizarán todavía más con el Z80: 


Ejercicio 5.1: Redáctese el programa de transferencia del Z80 en 
la forma del propuesto para el 6502, es decir, suponiendo que 
el registro de índice contiene un desplazamiento. Supóngase 
que los bloques fuente y destino se encuentran en la página 0 
y. por tanto, en las direcciones 0 a 256. Naturalmente, se 
supondrá también que el numero de elementos de cada bloque 
es suficientemente pequeño como para que no solapen. 


Ejercicio 5.2: Supóngase ahora que los bloques destino y fuente se 
encuentran en cualquier otro lugar de la memoria, aunque 
siempre dentro de la misma página. Escríbase de nuevo el 
programa con arreglo a esta nueva convención. (¿Hay alguna 
diferencia?, es decir, ¿desempeña la página 0 alguna función en 
el Z80?) 


RUTINA GENERAL DE TRANSFERENCIA DE 
BLOQUES (PARA MAS DE 256 ELEMENTOS) 


La distribución de registros y el mapa de la memoria se 
recogen en la figura 5.8. El programa es el siguiente: 


LD ВС, CUENTA NUMERO DE BYTES 

LD DE, HASTA DIRECCION DE DESTINO 

LD HL, DESDE DIRECCION DE PARTIDA 

LDIR TRANSFERENCIA DE TODOS 
LOS BYTES 


Memoria utilizada: 11 bytes. 
Tiempo: 21 ciclos por byte transferido. 
La primera instrucción es: 


LD ВС, CUENTA 


y carga el número de elementos que han de transferirse (un 
valor de 16 bits) en el par de registros BC. Las dos siguientes 
instrucciones inicializan el par de registros DE y HL, respecti- 
vamente: 


LD DE, HASTA 
LD HL, DESDE 


Por último, la cuarta instrucción: 


LDIR 


ejecuta la transferencia completa. 


DESTINO р DESDE 
H FUENTE 


REGISTROS 


HASTA 


Figura 5.8 
Transferencia de bloques: mapa 
de la memoria. 


MEMORIA 


LDIR es una instrucción automática de transferencia de blo- 
ques, con una potencia que este ejemplo hace obvia. LDIR da 
lugar a la siguiente secuencia: el contenido de la posición de 
memoria señalada por H y L se transfiere a la señalada por 
DE: (ЮЕ) = (НІ); а continuación se incrementa DE: 
DE = DE + 1; luego HL: HL — HL + 1; por fin, se decremen- 
ta ВС: ВС = BC- 1; si BC vale 0, la instrucción termina; 
en caso contrario, se ejecuta de nuevo. 

La utilidad y potencia de la instrucción LDIR debe ser ya 
| evidente sin necesidad de más comentarios. De forma similar, la 
\ búsqueda del carácter “asterisco” puede llevarse a cabo con otra 
| instrucción automática llamada CPIR, también especial del 780. 
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El programa sería: 


LDA, “*” 

LD BC, CUENTA 

LD HL, SERIE 

CPIR 

JR Z, ASTER 
NOAST --<- 


La primera instrucción carga el acumulador con el código 
del carácter asterisco. A continuación se inicializa el par de 
registros BC con la cuenta del número de palabras que hay que 
investigar dentro del bloque: 


LD ВС, CUENTA 


El par de registros H y L se lleva a la dirección de partida 
del bloque (SERIE). А continuación se ejecuta la instrucción 
automática: 2 


LD НЕ, SERIE 
CPIR 


La instrucción CPIR es una instrucción de comparación 
automática que compara el contentdo de la posición de memo- 
ria especificada por HL con el del acumulador. Si la compara- 
ción es negativa, la bandera Z del registro de estado se activa a 
1, el par HL se incrementa y el par BC se decrementa. La 
instrucción se repite hasta que el par BC vale 0 o hasta que la 
comparación es positiva. Por tanto, una vez ejecutada la ins- 
trucción CPIR, es preciso verificar la bandera Z, para determi- 
nar si el resultado ha sido o no positivo (en un caso extremo, la 
instrucción puede recorrer hasta 64K palabras sin haber obteni- 
do nada positivo); ésta es la finalidad de la última instrucción: 


JR 2, ASTER 


Ejercicio 5.3: Escribase de nuevo el programa anterior de manera 
que la búsqueda vaya hacia atrás. (Un consejo: utilicese la 
instrucción CPDR.) La transferencia del bloque debe conti- 
nuar hasta localizar el elemento “*”. 


Vamos a desarrollar ahora un programa que combine las 
caracteristicas de los dos anteriores, es decir, que ejecute la 
transferencia de un bloque desde la posición DESDE hasta la 
posición HASTA y que, a la vez, se detenga automáticamente si 


encuentra el carácter de escape “asterisco”. El programa sería 
así: 


LD BC, CUENTA 
LD HL, DESDE 
LD DE, HASTA 


LD А,“ DELIMITADOR 
(CARACTER DE 
ESCAPE) 
PROBAR CP (HL) COMPARAR CON 
EL CARACTER EN 
MEMORIA 
JR Z,FIN TERMINAR SI SE 
ENCUENTRA 
LDI TRANSFERIR EL 


CARACTER Y AC- 
TUALIZAR APUN- 
TADORES Y 
CUENTA 

JP | PE,PROBAR SEGUIR PROBAN- 
DO, SALVO QUE 
P/V INDIQUE 
BC = 0 


Las tres primeras instrucciones del programa se encargan de 
la habitual inicialización de los registros de cuenta y de los 
punteros fuente y destino: 


LD ВС, CUENTA 
LD HL, DESDE 
LD DE, HASTA 


El carácter asterisco se carga en el acumulador en la forma 
habitual, de manera que pueda compararse con el carácter leído 
en la posición de memoria: 


LD A, eee 
que es justamente lo que hace la siguiente instrucción: 


.PROBAR CP (HL) 


El resultado positivo o negativo de la comparación se averigua 
verificando la bandera Z, que valdrá 1 en el primer caso. De la 
verificación se encarga la instrucción 


JR Z,END 
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Viene ahora una instrucción automática de transferencia: 
LDI 


que transfiere el carácter y actualiza los punteros y la cuenta 
de una vez. LDI transfiere el contenido señalado por H y L a la 
posición de memoria señalada por D y E: (DE) = (HL). Tam- 
bién incrementa DE y HL: 


DE = DE + 1 
HL = HL + 1 


Para terminar, decrementa BC: BC = BC —1. La peculiaridad 
de la instrucción radica en que la bandera P/V se borra si BC 
se decrementa hasta “0”, y se pone a 1 en caso contrario. Esta 
situación la comprueba explícitamente la última instrucción del 
programa para determinar si es preciso salir o continuar: 


JP РЕ, PROBAR 


SUMA DE DOS BLOQUES 


Este nuevo programa sirve para sumar dos bloques elemen- 
to a elemento a partir de las direcciones BLQ1 y BLQ2. 
CUENTA es el número de elementos de cada uno de los 
bloques, que deben tener idéntica longitud. El programa, es: 


SUMBLQ LD ВО! 
LD  IY,BLQ2 
LD В, СОЕМТА 
XOR A 

BUCLE LD  A,X-0) 
ADC А, (IY +0) 


LD (ІХ,А 
DEC IX 
DEC IY 
DEC B 


JR NZ, BUCLE 


La figura 5.9 recoge la distribución de la memoria. El pro- 
grama es bastante sencillo: el número de elementos que deben 
sumarse se carga en el registro de contador B y los dos regis- 
tros de índice IX e IY se inicializan a sus valores BLQ1 y 
BLQ2: 


SUMBLQ LD IX, BLQ1 
LD IY, BLQ2 
LD B, CUENTA 


B| CONTADOR 


M 


22 BLK 2 
Suma de dos bloques: 
BLO1 = BLO1 + BLO2. MEMORIA 


Antes: de la primera suma se borra el bit de acarreo: 
XOR A 
Se carga el primer elemento en el acumulador: 


| BUCLE LD  A,{IX +0) 


A continuación se le suma el elemento correspondiente de 
BLQ2: 


ADC А, ПУ +0) 

Y el resultado раза а BLQI: 

LD (IX), A 

Los dos apuntadores X e Y se decrementan: 


DEC IX 
DEC IY 


Lo mismo que el registro contador: 
DEC B 


Si dicho contador no es 0, se repite el. bucle de suma: 


JR NZ,BUCLE 


Figura 5.10 
Organización de la memoria 
para transferencia de bioques. 


Resumen 
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Ejercicio 5.4: ¿Sabría utilizar el programa anterior para ejecutar 
una suma de 32 bits? 


Ejercicio 5.5: ¿Y una suma de 64 bits? 


Ejercicio 5.6: Modifique el programa de manera que el resultado 
pase а un nuevo bloque que empiece en la dirección BIQ3. 


Ejercicio 5.7: Modifique el programa para que ejecute una resta 
en lugar de una suma. 


Ejercicio 5.8: Modifique el programa en el sentido de que BLQ1 y 
BLQ?2 constituyen las respectivas partes superiores de cada 
uno de los bloques en lugar de las inferiores (véase figura 
5.10). 


DESDE 


CUENTAS ч BLOQUE FUENTE 


ARS TRANSFERENCIA 


ELEMENTO 
HASTA — | 


x 


CONTADOR 


Hemos hecho una descripción completa de los modos de 
direccionamiento y hemos visto que el Z80 dispone de numero- 
sos mecanismos y de modos de direccionamiento específicos. 
Los diversos programas de aplicaciones han servido para de- 
mostrar el valor de tales mecanismos. Quien desee aprender a 
programar con eficacia el Z80 deberá llegar a entenderlos per- 
fectamente. A partir de ahora se utilizarán ampliamente en 
todos los programas del libro. 


енен 


Ejercicio 5.9: Escriba un programa раға sumar los 10 primeros 
bytes de una tabla almacenada en la posición “ВАЗЕ”; el 
resultado tendrá 16 bits (se trata de un cálculo de total de 
control). 


Ejercicio 5.10: ¿Podría resolver el mismo problema sin utilizar 
indexación? 


Ejercicio 5.11: Invierta el orden de los 10 bytes de la tabla y 
almacene el resultado en la dirección “REVER”. 


Ejercicio 5.12: Busque el elemento mayor de esa misma tabla y 
almacénelo en la dirección de memoria “GRANDE”. 


Ejercicio 5.13: Sume los elementos correspondientes de las tres 
tablas, cuyas bases son ВАЗЕ1, BASE2 y BASE3. La longi- 
tud de las tablas se almacena en la dirección “LONGITUD”. 
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а — ae : 
—— циан рН ЕЕ 


us ЖШ 
duo ШЕ 
LR. 
LIE 
Ge ти 
© Жж 
LEE J 
w з 
ж ш 
ж ж 
ж am 
ru 
ж 


Técnicas 


de entrada/salida 


Introducción 


Entrada/salida 


Hasta el momento hemos aprendido a intercambiar infor- 
mación entre la memoria y los diversos registros del procesa- 
dor, a manipular dichos registros y a mover datos. Ahora 
debemos aprender a comunicarnos con el mundo exterior, que 
es justamente lo que se entiende por entrada/salida. 

La entrada es la obtención de datos en los periféricos 
externos (teclado, discos o sensores físicos). Salida es la entrega 
de datos por parte del microprocesador o la memoria a disposi- 
tivos externos, como una impresora, una pantalla, un disco о 
sensores y relés. 

En una primera etapa aprenderemos a realizar las operacio- 
nes de entrada/salida que exigen los dispositivos más comunes. 
En segundo lugar aprenderemos a manejar varios de esos 
dispositivos simultáneamente, es decir, a organizarlos; en esta 
segunda fase centraremos la discusión más en concreto en el 
uso del muestreo frente a las interrupciones. 


Vamos antes de nada a aprender a percibir y a generar 
señales sencillas o impulsos. A continuación presentaremos 
algunas técnicas para garantizar o medir una sincronización 


— 


H 


A 
correcta. Una vez afianzada esta base, podremos pasar a estu- 
diar técnicas de entrada/salida más complejas, como las transfe- 
rencias a alta velocidad en serie y en paralelo. 


INSTRUCCIONES DE ENTRADA/SALIDA DEL Z80 


El Z80 dispone de una serie de instrucciones especiales para 
este cometido. La mayor parte de los microprocesadores de 8 
bits carecen de ellas y controlan los dispositivos de entrada/sali- 
da con instrucciones generales. También el 8080 dispone de 
instrucciones especiales, aunque el Z80 cuenta con algunas 
nuevas, que describiremos más detalladamente para facilitar la 
comprensión de los programas que analizaremos en este capítu- 
lo. 

Las instrucciones básicas de entrada y salida son, respectiva- 
mente: IN A, (n) y OUT (п), A, ambas heredadas del 8080. Leen 
o escriben, según, un byte entre la puerta seleccionada y el 
acumulador. El proceso de direccionamiento es tal que la 
dirección del dispositivo de E/S “п” se deposita en las líneas А0 
a A7 del bus de direcciones, y el contenido del acumulador pasa 
a las líneas A8 a A15. Si sólo se direccionan 256 dispositivos, 
puede ser necesario llevar a 0 explícitamente el contenido del 
acumulador en el caso de que las líneas de dirección A8 a A15 
puedan ser decodificadas por un dispositivo de E/S. En los 
sencillos ejemplos que veremos a continuación supondremos 
que hay menos de 256 dispositivos y que no están conectados a 
las direcciones que van de A8 a A15, de manera que no será 
preciso hacer O el contenido del acumulador explícitamente 
antes de emplear, por ejemplo, la instrucción IN. 

Una instrucción de entrada especial —IN r,(C)— permite 
utilizar el contenido del registro C como dirección del dispositi- 
vo de E/S. Al utilizarla, el contenido del registro B proporciona 
automáticamente la parte superior de la dirección (A8 a A15). 
El registro especificado r se carga a partir de la dirección dada; 
“r” puede ser uno cualquiera de los siete registros de tipo 
general. 


GENERACION DE UNA SENAL 


En el caso más sencillo, es el ordenador el que desconecta (o 
conecta) los dispositivos de salida. Para modificar el estado de 
uno de tales dispositivos, el programador no tiene más que 
cambiar el nivel de un “0” lógico a un “1” lógico, o de “1” a 
“0”. Supongamos que al bit “0” de un registro llamado *OUTI" 


está conectado un relé externo; para conectarlo no tenemos 
más que escribir un “1” en la posición de bit adecuada del 
registro. Supongamos aquí que OUTI representa la dirección 
de este registro de salida dentro de nuestro sistema; el siguiente 
programa conectaría el relé: 


CONECTO LD А, 00000001B CARGAR DISPO- 
SICION EN A 

OUT (OUT1),A ENVIARLA AL 
DISPOSITIVO 


donde OUT es la instrucción de salida. 

Hemos supuesto que la situación de los siete bits restantes 
del registro OUTI es irrelevante, aunque las cosas no suelen ser 
asi, porque esos bits pueden estar conectados a otros relés; por 
tanto, vamos a mejorar tan elemental programa. Lo que hare- 
mos ahora es conectar el relé, pero sin alterar el estado de los 
demás bits del registro. Supongamos que es posible leer y 
escribir el contenido del mismo. La versión mejorada sería 
como sigue: 


CONECTO IN A, (OUTI) LEER EL CONTENI- 
DO DE OUTI 
OR 000000018 FORZAR ВІТ “0” А 
“1” EN A 
OUT (OUTI) A 


El programa empieza por leer el contenido de la posición 
ОСТІ, al que a continuación somete a la operación OR, 
que pone a “1” el bit de posición O sin afectar al resto (para 
más detalles sobre la operación OR, consúltese el capitulo 4). 
La figura 6.1 recoge el proceso. 


ANTES DESPUES 


RELE 


Figura 6.1 
Conexión de un relé. OUT 1 Out! 


IMPULSOS 


La generación de un impulso se lleva a cabo de la misma 
forma que acabamos de ver para el nivel: el bit de salida se 
pasa primero a 1, y a continuación otra vez a 0, lo que da lugar 
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Figura 6.2 
impulso programado. 


a un impulso, como ilustra la figura 6.2. No obstante, en esta 
ocasión hay que resolver otro problema: el impulso debe durar 
un tiempo determinado. Por tanto, vamos a estudiar antes de 
nada cómo se produce un retraso en el ordenador. 


REGISTRO 
CPU DE LA PUERTA 


DE SALIDA SEÑAL 


-—— МИ5ЕС — 


0--1 t=- 0 


PROGRAMA: ELEGIR PUERTA DE SALIDA 

CARGAR EL REGISTRO DE LA PUERTA DE SALIDA CON LA DISPOSICION 
DE ESPERA (BUCLE DURANTE N HUSEG) 

CARGAR CERO EN LA PUERTA DE SALIDA 

RETORNAR 


PRODUCCION Y MEDIDA DE RETARDOS 


El retardo puede producirse mediante los soportes lógico o 
físico. Aquí aprenderemos a crearlo en el programa, y más 
adelante ya estudiaremos la manera de obtenerlo con un conta- 
dor físico o sincronizador de intervalos programable, como 
suele llamarse (РТТ). 

Los retardos programados se generan contando; el registro 
contador se carga con un valor que se va decrementando. El 
programa describe un bucle y reduce una y otra vez el conteni- 
do del contador hasta que llega a “0”; el tiempo consumido por 
esta serie de operaciones será el retardo buscado. Como ejem- 
plo, vamos a generar un retardo de 82 ciclos de reloj: 


RETARDO LD А, 5 A ES EL CONTA- 
DOR 
BUCLE DEC А DECREMENTO 
JR NZ, BUCLE BUCLE COM- 
PROBACION 


El programa carga en A el valor 5; la siguiente instrucción 
decrementa A, y la otra provoca un salto a BUCLE mientras A 
no valga “0”; cuando alcanza este valor, el programa sale del 
bucle y ejecuta la instrucción que sigue. La lógica del programa 
es sencilla, y se ilustra en el diagrama de flujo de la figura 6.3. 


Figura 6.3 
Diagrama de flujo del retardo 
básico. 


CONTADOR = VALOR 
CONTADOR 
DE DECREMENTO 


SALIDA 


Vamos ahora a calcular el retardo real provocado por el 
programa. En el capítulo 4 encontraremos el tiempo consumido 
por cada una de las instrucciones. 

LD en modo inmediato necesita 7 ciclos; DEC emplea 4; 
por último, JR consume 12 ciclos, salvo en la última repetición, 
que sólo emplea 7. En efecto, al consultar la tabla correspon- 
diente a JR vemos que hay dos posibilidades: si no hay salto, la 
instrucción termina en 7 ciclos, y si lo hay y debe recorrer el 
bucle, necesita 12. 

Así que se consumen 7 ciclos en la primera instrucción, 
más 16 multiplicado por el número de veces que se repite el 
bucle en las dos siguientes menos 5, por el último salto que no 
se produce: 


Retardo = 7 + 16 x 5 — 5 = 82 ciclos. 


Si el ciclo es de 0.5 microsegundos, el retardo sería de 41 
microsegundos. 

El bucle de retardo que acabamos de examinar se emplea en 
casi todos los programas de entrada/salida; conviene, por tanto, 
entenderlo bien. Resuelva los dos ejercicios siguientes: 


Ejercicio 6.1: ¿Cuáles son los retardos máximo y mínimo que 
pueden conseguirse con estas tres instrucciones? 


Ejercicio 6.2: Modifiquese el programa para obtener un retardo de 
unos 100 microsegundos. 
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ПЕ! 


Para conseguir un retardo más largo, una solución sencilla 
es añadir más instrucciones al programa antes de DEC; la 
instrucción más adecuada es NOP, que mantiene al procesador 
inactivo durante cuatro ciclos. 


RETARDOS SUPERIORES 


Para conseguir retardos mayores por medio del soporte 
lógico es necesario utilizar un contador más amplio; si éste 
tiene 16 bits, puede alojarse en un par de registros. Para 
simplificar, supongamos que la cuenta menor es “0”. El byte 
inferior se carga con “0”, y a continuación la superior recorre 
un bucle de decrementación. Como la primera operación reduc- 
tora da lugar a 00 > FF y no afecta a la bandera Z cuando se 
decrementa a “0”, el byte superior del contador se decrementará 
en 1. Cuando este byte llega al valor “0”, el programa termina. | 
Si hace falta una precisión mayor еп la producción del retardo, | 
la cuenta inferior puede llevar un valor no nulo. Еп este caso 
escribiriamos el programa tal como hemos explicado y añadi- 
ríamos al final el de tres líneas estudiado en el apartado | 
anterior. | 

He aquí un programa de retardo de 24 bits: 


КЕТ24 LD B, CUENTAS CONTADOR SU- 
PERIOR (3 BITS) | 
КЕТІ6 LD D, — 1 | 
BUCLEA LD HL, CUENTAI CONTADOR IN- 
FERIOR 
BUCLEB ADD HL, DE DECREMENTO | 
HL 
JR C, BUCLEB SIGUE HASTA 
HACERLO NU- 
LO 
DJNZ BUCLEA DECREMENTO 


DE B Y SALTO 


Obsérvese que DE se carga con “— 1", y se usa para 
decrementar el contador de 16 bits HL. | 

Naturalmente, pueden conseguirse retardos todavia mayores 
con más de tres palabras. El funcionamiento es análogo al de 
un cuentakilómetros de coche: cuando la rueda de la derecha 
pasa de 9 a 0, la siguiente se incrementa en 1. El cómputo con 
unidades discretas se basa siempre en este principio general. 

El principal inconveniente de esta técnica es que el micro- 
procesador pasa cientos de milisegundos y hasta de segundos 


sin hacer nada. Si el ordenador no tiene ningún otro trabajo 
que realizar, esto no tiene importancia, pero, en general, deberá 
estar disponible para ejecutar otras tareas, y por eso los retar- 
dos muy largos no suelen producirse mediante el soporte lógico. 
De hecho, hasta los retardos más breves pueden ser problemáti- 
cos si el sistema debe garantizar una respuesta dentro de un 
plazo de tiempo limitado en algunas situaciones. En todos estos 
casos hay que recurrir a los retrasos de soporte físico. Por otra 
parte, cuando se trabaja con interrupciones, la detención por 
éstas de un bucle acarrearía la pérdida de la exactitud de la 
sincronización. 


Ejercicio 6.3: Escriba un programa para producir un retardo de 
100 ms (típico de un teletipo). 


RETARDOS EN HARDWARE 


Estos retardos se crean empleando un sincronizador de 
intervalos programable o cronómetro. Se carga un valor en el 
registro del cronómetro, y éste se encarga automáticamente de 
decrementar el contador. Por lo general, el programador puede 
ajustar o escoger el período; cuando llega a “0”, el dispositivo 
suele enviar una interrupción al microprocesador; también 
puede activar un bit de estado inspeccionado periódicamente 
por el ordenador. El empleo de interrupciones se explicará más 
adelante en este mismo capítulo. 

El cronómetro puede también partir de “0” y contar la 
duración de la señal o contar el número de impulsos recibido. 
Cuando funciona como cronómetro de intervalos, se dice que 
opera en modo paso a paso. Si cuenta impulsos, el modo se 
describe como contador de impulsos. Algunos dispositivos sincro- 
nizadores incluyen varios registros y recursos opcionales que el 
programador puede elegir. 


RECEPCION DE IMPULSOS 


El problema que plantea la recepción de impulsos es inverso 
al de su producción, pero con una dificultad adicional: mientras 
que la producción de un impulso tiene lugar bajo control del 
programa, su recepción es asíncrona respecto al mismo. Para 
detectarlo se utilizan dos técnicas: el muestreo de dispositivos y 
las interrupciones; de éstas hablaremos más adelante. 

Detengámonos ahora en la técnica de muestreo, en virtud de 
la cual el programa lee continuamente el valor de un registro 
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dado de entrada y comprueba en el mismo un bit determinado 
(por ejemplo, el 0). Cada vez que se recibe un impulso, el bit 
adopta el valor “1”; el programa vigila el bit O hasta que pasa a 
“1”, y en ese momento detecta un impulso. El programa 
necesario para ello es: 


MUESTRA IN A,(ENTRADA) LEER REGISTRO 


DE ENTRADA 
ON BIT ОА COMPROBAR EL 
BIT 0 
JR . Z, MUESTRA SEGUIR LA MUES- 
TRA, SI ES 0 


Supongamos ahora que la línea de entrada vale normalmen- 
te “1” y que deseamos detectar la presencia de “0”. Es el 
procedimiento habitual para detectar un bit de ARRANQUE 
cuando se controla una línea conectada a un teletipo; el 
programa sería: 


MUESTRA IN  A,(ENTRADA) LEER REGIS- 
TRO DE EN- 
TRADA 
BIT ОА ACTIVAR LA 
BANDERA Z 
IR NZ,_MUESTRA COMPROBAR 
SI SE HA IN- 
VERTIDO 
ARRANQUE ... 


CONTROL DE LA DURACION 


La duración de un impulso de entrada puede controlarse de 
la misma forma que se calcula la de uno de salida, y también en 
este caso puede recurrirse al soporte lógico o al físico. En el 
primer caso, lo normal es incrementar un contador de 1 en 1; 
para ello, el programa recorre un mismo bucle mientras hay 
impulso; cuando éste termina, su duración se calcula a partir 
del valor del registro contador. El programa es el siguiente: 


DURACION LD В,0 BORRAR CON- 
TADOR 
OTRA IN А (ENTRADA) LEER ENTRADA 
ВТ ОА COMPROBAR 
BIT 0 


JR Z, OTRA ESPERAR UN “1” 
B 


MAYOR INC INCREMENTAR 
CONTADOR 
IN A, (ENTRADA) COMPROBAR 
: BIT 0 


BIT ОА 
JR NZ, MAYOR ESPERAR UN “0” 


Naturalmente, se supone que la duración máxima del impul- 
so no provocará el desbordamiento del registro B, porque en 
ese caso habría que modificar el programa, para tenerlo en 
cuenta (de no hacerlo así, se produciría un error). 

Como ya sabemos producir y detectar impulsos, podemos 
pasar a recibir o transferir cantidades superiores de datos. En 
este nuevo problema cabe distinguir dos situaciones: transferen- 
cia en serie y transferencia en paralelo. Tras su solución empe- 
zaremos a trabajar con dispositivos de entrada/salida reales. 


Transferencia de palabras en paralelo 


Partimos de la suposición de queen la dirección "ENTRADA" 
(véase figura 6.4) disponemos en paralelo de 8 bits del dato a 
transferir. El microprocesador debe leer el byte situado en esa 
posición siempre. que la palabra de estado le indique que es 
válida. Supondremos que dicha información de estado se en- 
cuentra en el bit 7 de la dirección “ESTADO”. Vamos a escribir 
un programa que sea capaz de leer y reservar automáticamente 
cada una de las palabras de dato que reciba. Para simplificar 
las cosas, también supondremos que conocemos de antemano el 
nümero de palabras que deben leerse, almacenado en la posi- 
ción "CUENTA". Si no dispusiésemos de tal información, 
tendríamos que buscar un carácter de ruptura, o de borrado, o 
quizá un asterisco "x", operación que ya sabemos hacer. 

El diagrama de flujo aparece en la figura 6.5 y es muy 
sencillo. Se vigila la información de estado hasta que vale “1”, 
lo que indica que una palabra está lista. Cuando ocurre esto, se 
lee y se guarda en una posición de memoria adecuada; a 
continuación se decrementa el contador y se comprueba si ha 
llegado a “0”, lo que significaría el fin del programa; en caso 
contrario, se lee una nueva palabra. He aquí un programa 
sencillo que desarrolla el algoritmo descrito: 
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PARAL LD А, (CUENTA) 
LD B,A 

PROBAR IN А, (ESTADO) 
BIT 7,A 
JR Z, PROBAR 
IN А, (ENTRADA) 
PUSH АЕ 
DEC B 
JR NZ, PROBAR 


CUENTA 


ESTADO 


ENTRADA 


Figura 6.4 
Transferencia de palabras en 
paralelo: memoria. 
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Ац 


РР 
=> > 


LEER CUENTA 
EN A 

B ES EL CON- 
TADOR 
COMPROBAR 
SI “DATO LIS- 
TO” ES CIER- 
TO 

BIT 7 ES “1”, SI 
EL DATO ES- 
TA LISTO 

¿ES VALIDO 
EL DATO? 
LEER EL DA- 
TO 

GUARDAR EL 
DATO EN LA 
PILA 
DECREMEN- 
TAR CUENTA 
REPETIR HAS- 
TA CERO 


VALIDO 


DISPOSITIVO 
DE E/S 


Se supone que la bandera “dato listo” se borra automática- 
mente al leer ESTADO. 

Las dos primeras instrucciones inicializan el registro conta- 
dor B: 


PARAL LD A,(CUENTA) 
LD ВА 


Obsérvese que no hay forma fácil de cargar sólo B a partir 


de la memoria; es preciso cargar Á y transferir su contenido a 
B o cargar B y C simultáneamente. 


MUESTREO O SOLICITUD DE SERVICIO 


LEER CUENTA 


¿PALABRA LISTA? 


SI 


TRANSFERIR 
PALABRA 


DECREMENTAR 
CONTADOR 


Figura 6.5 
Transferencia de palabras en 
paralelo; diagrama de flujo. SALIDA 


Las tres instrucciones siguientes del programa leen la infor- 
mación de estado y recorren un bucle mientras el bit siete del 
registro de estado vale, “0” (se trata del bit del signo, es decir, 


del Bit N). 
IN A, ESTADO 
BIT 7,A “IN” NO ACTIVA LAS BANDE- 
RAS 


JR Z, PROBAR 
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Cuando JR no se realiza, el dato es válido y puede leerse: 
IN A,(ENTRADA) 


La palabra ya se ha leído en la dirección de (ENTRADA) en 
que estaba, y ahora debe guardarse. Suponiendo que hay 
suficiente sitio en la pila, podemos usar la instrucción 


PUSH AF 


que archiva A (y F) en la pila. Si ésta está completa, o si hay 
que transferir un número elevado de palabras, no podremos 
llevar a cabo la operación de PUSH, y tendremos que transferir 
los datos a un área de memoria designada al efecto mediante 
una instrucción indexada, por ejemplo. Lo malo es que ello 
exigiría una instrucción adicional para incrementar o decremen- 
tar el registro de índice. La operación PUSH es más rápida 
(sólo 11 ciclos de reloj). 

La palabra dato ya está leída y archivada. Sólo nos queda 
decrementar el contador de palabra y comprobar si ha termina- 
do: 


DEC B 
JR NZ, PROBAR 


Ese programa de nueva instrucción podría considerarse de 
referencia. Se llama así a un programa cuidadosamente optimi- 
zado pensado para poner a prueba el rendimiento de un 
procesador ante una situación dada. La transferencia en parale- 
lo es una de esas situaciones típicas, y el programa en cuestión 
está diseñado de manera que alcance la eficacia y la velocidad 
máximas. Vamos ahora a calcular la velocidad máxima de 
transferencia del mismo. Supondremos que CUENTA está en 
memoria. La duración de cada una de las instrucciones se 
determina consultando las tablas del capítulo 4; el resultado es 
el siguiente: 


PARAL LD А, (CUENTA) 13 
LD B, A 4 
PROBAR IN A, (ESTADO) 11 
BIT 7, À 8 
JR Z, PROBAR 7/12 
IN A, (ENTRADA) 11 
PUSH АЕ 11 
DEC B 4 


JR NZ, PROBAR 7/12 


El tiempo mínimo de ejecución se obtiene suponiendo que 
cada vez que comprobamos ESTADO aparece un dato; en otras 
palabras, suponiendo que el primer salto JP falla siempre. En 
tal caso, el tiempo sería: 


13+4+ (11+8+7+11 +4 + 12) * CUENTA 


Despreciando los primeros 17 ciclos necesarios para iniciali- 
zar el registro contador, el tiempo necesario para transferir una 
palabra es de 64 ciclos de reloj, equivalentes a 32 microsegun- 
dos con un reloj de 2 MHz. 

Por tanto, la cadencia de transferencia máxima es: 


1 
32(10 7^) 
Ejercicio 6.4: Supongamos que han de transferirse más de 256 
palabras. Modifique el programa en consecuencia y determine 


la influencia de la ampliación en la velocidad máxima de 
transferencia. 


= ЗІК por segundo 


Ejercicio 6.5: Modifique el programa para tratar de aumentar su 
velocidad; siga uno de estos procedimientos: 


1. Emplear JP en lugar de JR. 
2. Utilizar DJNZ. 
3. Utilizar INI o IND. 


¿Era el programa verdaderamente óptimo? 


Ya sabemos hacer transferencia en paralelo a alta velocidad. 
Pasemos ahora a considerar un caso más complejo. 


Transferencia de bits en serie 


Se llama entrada en serie a la que tiene lugar de manera 
que los bits pasan, en secuencia, uno tras otro. Si llegan a 
intervalos regulares, se habla de transmisión síncrona; es asín- 
crona si la entrada se produce al azar, en forma de grupos de 
datos. Vamos a desarrollar un programa que funcione en los 
dos casos. El principio de la captación de datos secuenciales es 
sencillo: se comprueba una línea de entrada, que supondremos 
es la linea 0; cuando se detecta un bit en la misma, se lee y se 
desplaza a un registro de almacenamiento; una vez reunidos 8 
bits, el byte resultante se guarda en memoria, y se pasa a montar 
el siguiente. Para simplificar las cosas, supondremos que se sabe 
de antemano el número de bytes que va a recibirse; en caso 
contrario podríamos verificar la llegada de un carácter de | 
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ruptura y detener la transferencia en ese momento, cosa que ya 
sabemos hacer. El diagrama de flujo del programa aparece en la 
figura 6.6; el programa es el siguiente: 


Figura 6.6 
Transferencia de bits en serie; 
diagrama de flujo. 


SERIE 


BUCLE 


MUESTREO O SOLICITUD DE SERVICIO 


LECTURA CONTADOR 
DE PALABRAS 


ALMACENAR BIT 
INCREMENTAR CONTADOR 


¿PALABRA TERMINADA? 


ALMACENAR PALABRA 
REINICIAR CONTADOR DE BITS 
DECREMENTAR CONTADOR 
DE PALABRAS 


¿CONTADOR 
DE PALABRAS = 0? 


BORRAR  PALA- 
BRA DE ENTRA- 


CARGAR B CON 
EL CONTADOR 


BIT 7 ES EL ESTA- 
DO Y BIT 0 EL 


ESPERAR UN “1” 


TERMINADO 
LD C,0 
DA 
LD А, (CUENTA) 
DE BYTE 
LD B,A 
IN А, ENTRADA LEER PUERTA 
BIT 7, A 
DATO 
JR Z, BUCLE 
SRL 


DESPLAZAR EL 
BIT DE DATO AL 
ACARREO 


RL С GUARDAR EN- 
TRADA B EN C 

JR NC, BUCLE SEGUIR HASTA 
QUE ENTREN 8 
BITS 

PUSH BC GUARDAR LA PA- 
LABRA EN LA PI- 
LA 

LD C, 01H LLEVAR A 0 EL 
BIT MARCADOR 

DEC B DECREMENTAR 
ЕГ CONTADOR 
DE BYTE 

JR NZ, BUCLE MONTAR LA PA- 
LABRA SIGUIEN- 
TE 


El programa está diseñado para conseguir la máxima efica- 
cia y utiliza técnicas nuevas que explicaremos a continuación 
(véase figura 6.7). 


CUENTA 


EN 


© PAE 


ESTADO 
0 


RELOJ 


ENTRADA 
DE DATOS 
EN SERIE 
ENTRADA 


Figura 6.7 
Serie-Paralelo: Los registros. 


Las convenciones adoptadas son las siguientes: se supone 
que la posición de memoria CUENTA alberga el número de 
palabras que va a transferirse; el registro С se utiliza para 
reunir los 8 bits consecutivos que llegan; la dirección ENTRA- 
DA corresponde al registro de entrada; se supone que el bit 7 
de este registro es una bandera de estado o un bit de reloj; el 
dato no es válido si es “0” y lo es si vale "1". El dato 
propiamente dicho se supone que aparece en el bit O de la 
misma dirección. En muchos casos, la información de estado no 
se encuentra en el registro de datos, sino en otros; por tanto, 
modificar el programa debe resultar sencillo. También se supo- 
ne que el primer bit del dato recibido es siempre “1”, lo que 
indica que a continuación viene el dato real. Si esto no fuese 
así, veremos más adelante una modificación obvia para tenerlo 
en cuenta. El programa responde exactamente al diagrama de 
flujo de la figura 6.6. Las primeras líneas del mismo realizan un 
bucle de espera que comprueba si hay un bit dispuesto; para 
determinar este hecho se lee el registro de entrada y a continua- 
ción se comprueba el bit 0 (2); si éste vale “0”, la instrucción JR 
se efectuará, y el programa entrará en el bucle. Cuando el bit de 
estado (o de reloj) indica certeza (“1”), el salto no se efectúa, y 
pasa a ejecutarse la siguiente instrucción. 

Esta secuencia inicial corresponde a la flecha 1 de la figu- 
га 6.7. 

En este punto el acumulador contiene un “1” en el bit 7 y el 
bit correspondiente al dato real en el bit 0. El primer bit que 
llega es “1”, pero los siguientes pueden valer “0” o “1”. Ahora 
nos interesa conservar el bit de dato recogido en la posición 0; 
la instrucción 


SRL A 


desplaza el contenido del acumulador una posición a la dere- 
cha, lo que hace que el bit derecho de A, que es el correspon- 
diente al dato, pase al bit de acarreo. Ahora lo mantendremos 
en el registro C (el proceso está representado por las flechas 2 y 
3 de la figura 6.7): 


RL C 


El efecto de esta instrucción es leer el bit de acarreo en la 
posición derecha de C. Al mismo tiempo, el bit de la izquierda 
de С pasa al bit de acarreo (si tiene alguna duda sobre la 
operación de rotación, consulte el capítulo 4). 


Es importante recordar que la rotación con acarreo guarda 
el bit de acarreo, en este caso en la posición derecha, y también 
lo repone con el valor del bit 7 (o bit 0). 

En nuestro ejemplo pasará un “0” al acarreo. La siguiente 
instrucción : 


JR МС, BUCLE 


comprueba el acarreo y bifurca el programa a la dirección 
BUCLE, si aquél es “0”; es nuestro contador de bit automático. 
Es fácil comprobar que, como resultado de la primera aplica- 
ción de RL, € contendrá “00000001”. Ocho desplazamientos 
más tarde, el “1” pasará, por fin, al bit de acarreo y detendrá la 
bifurcación. Es un método ingenioso de crear un contador 
automático de bucle sin desperdiciar una instrucción para 
decrementar el contenido del registro de índice; la técnica se 
emplea aquí para acortar el programa y mejorar su rendi- 
miento. 

Cuando deje de actuar JR NC, en C se habrán reunido 8 
bits, valor que deberá conservarse en la memoria. De ello se 
encarga la siguiente instrucción (flecha 4 de la figura 6.7): 


PUSH BC 


Los contenidos de B y C pasan a la pila; esta operación 
sólo es posible si la mencionada pila tiene sitio suficiente; 
suponiendo que tal condición se cumple, es la forma más rápida 
de guardar una palabra en memoria, a pesar de que también se 
guarda un registro innecesario (B); el apuntador de la pila se 
actualiza automáticamente. Si no queremos introducir la pala- 
bra en la pila, tendremos que utilizar una instrucción más para 
actualizar el puntero de la memoria. También podríamos 
emplear una dirección indexada equivalente, pero eso supondria 
decrementar o incrementar el índice, Operación que consume 
todavía más tiempo. 

Una vez almacenada la primera palabra del dato, ya no hay 
garantía de que el primer bit vuelva a ser "1"; por tanto, es 
preciso reiniciar el contenido a “00000001” para poder seguir 
utilizándolo como contador de bit; de ello se encarga la 
instrucción 


LD сон 


Por último, decrementaremos el contador de palabras, ya 
que se ha montado una, y comprobaremos si hemos llegado al 
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término de la transferencia. Para ello sirven las dos instruccio- 
nes que siguen: 


РЕС В 
JR NZ, BUCLE 


El programa está diseñado en función de la velocidad, para 
que pueda captar una corriente rápida de bits de entrada. 
Cuando termina es aconsejable leer inmediatamente las pala- 
bras guardadas en la pila y transferirlas a cualquier otra 
posición de la memoria; en el capítulo 2 ya aprendimos a 
realizar una transferencia de bloques de esa naturaleza. 


Ejercicio 6.6: Calcúlese la máxima velocidad de lectura de bits en 
serie por el programa. Para ello se consulta en las tablas del 
capítulo 4 los ciclos que consume cada instrucción; para 
determinar el tiempo consumido por un bucle no hay más que 
multiplicar la duración total del mismo expresada en microse- 
gundos por el número de veces que debe ejecutarse. A efectos 
de calcular la velocidad máxima, supóngase que hay un bit de 
dato listo cada vez que se detecta la posición de entrada. 


Este programa es más difícil de entender que los anteriores. 
Vamos, pues, a analizarlo de nuevo (consulte la figura 6.6) más 
en detalle, estudiando posibles alternativas. 

De vez en cuando llega un bit de dato a la posición O de 
"ENTRADA"; puede haber, por ejemplo, tres “1” seguidos; por 
tanto, es preciso diferenciar los bits sucesivos. Esta es justamente 
la función de la señal de “reloj”. 

La señal de reloj (o ESTADO) dice que el bit de entrada es 
en este momento válido. Así pues, antes de leer un bit debemos 
verificar el de estado; si es “0”, esperaremos; si es “1”, significa 
que el bit de dato es correcto. 

Hemos supuesto que la señal de estado está conectada al bit 
7 del registro ENTRADA. 


Ejercicio 6.7: ¿Puede explicar por qué se emplea el bit 7 para el 
estado y el 0 para el dato? ¿Tiene la elección alguna impor- 
tancia? 


Una vez captado un bit, es necesario guardarlo en una 
posición segura, y a continuación desplazarlo a la izquierda, 
para dejar sitio al bit siguiente. 

Por desgracia, el acumulador se está usando para leer y 
comprobar los datos y el estado; por tanto, si lo utilizásemos 
también para acumular el dato, el bit de la posición 7 podria 
ser borrado por el bit de estado. 


Ejercicio 6.8: ¿Podría sugerir alguna forma de comprobar el bit de 
estado sin borrar el contenido del acumulador? ¿Sería con 
alguna instrucción especial? Si tal cosa fuese factible, ¿serviria 
el acumulador para conservar los bits que llegan en sucesión? 
¿Aumentaría la velocidad con un salto automatizado? 


Ejercicio 6.9: Vuelva a escribir el programa usando el acumulador 
para almacenar los bits de entrada. Compárelo con el anterior 
en términos de velocidad y número de instrucciones. 


Vamos a examinar otras dos posibles variantes. En nuestro 
ejemplo hemos supuesto que el primer bit sería una señal 
especial y valdría siempre “1”; pero, en general, puede ser 
cualquiera. 


Ejercicio 6.10: Modifique el programa anterior suponiendo que el 
primer bit es un dato válido que no debe descartarse y que 
puede valer tanto “0” como “1”. Un consejo: el contador de 
bits seguirá funcionando correctamente si se inicializa el valor 
adecuado. 


Para ganar tiempo, hemos guardado la palabra formada en 
la pila; pero, naturalmente, podríamos archivarla en cualquier 
posición especificada de la memoria. 


Ejercicio 6.11: Modifique el programa anterior en el sentido de 
guardar la palabra reunida en el área de memoria que empieza 
en BASE. 


Ejercicio 6.12: Modifique el programa anterior para que la 
transferencia se detenga en cuanto se detecte el carácter “S” en 
la corriente de entrada. 


LA OPCION DEL SOPORTE FISICO 


La mayor parte de los algoritmos de entrada/salida pueden 
ejecutarse en el soporte físico por medio de una pastilla llamada 
UART, que acumula automáticamente los bits. No obstante, si 
se desea reducir el gasto de componentes, puede utilizarse el 
programa que acabamos de ver o una variante del mismo. 


Ejercicio 6.13: Modifique el programa en el sentido de suponer 
que el dato queda disponible en el bit 0 de la posición 
ENTRADA y la información de estado en el bit 0 de la 
dirección ENTRADA + 1. 
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Resumen básico de E/S 


Ya sabemos realizar operaciones elementales de entrada/sali- 
da y manipular corrientes de datos en paralelo o en serie. 
Ahora ya podemos pensar en comunicarnos con dispositivos 
reales de entrada/salida. 


Comunicación con dispositivos de entrada/salida 


Para intercambiar datos con dispositivos de entrada/salida, 
el primer paso es asegurarse de que existen datos, si queremos 
leerlos, o de que el dispositivo está listo para aceptarlos, en el 
caso contrario. Para ello se recurre a dos técnicas: el acopla- 
miento y la interrupción; empecemos por la primera. 


ACOPLAMIENTO 


El acoplamiento es la técnica empleada habitualmente para 
establecer comunicación entre dos dispositivos asíncronos, es 
decir, no sincronizados. Si, por ejemplo, queremos enviar una 
palabra a una impresora en paralelo, antes de nada debemos 
asegurarnos de que está disponible la memoria auxiliar de 
entrada de dicho dispositivo. Así que preguntaremos a la 
impresora: “¡estás lista?”, y ella responderá “sí” o “no”. Si no 
está lista, habrá que esperar; si lo está, enviaremos los datos 


(véase figura 6.8). 
Mes 
DE 
ESTADO 


REGISTRO 
DE 
SALIDA 
Figura 6.8 


Acoplamiento (salida). PASTILA DE E/S 


¿LISTO? 
(LEE 
ESTADO) 


DISPOSITIVO 
DE 
SALIDA 


Y viceversa, antes de leer los datos procedentes de un 
dispositivo de entrada verificaremos si son válidos. Preguntare- 
mos; “¿es válido el dato?”, a lo que el dispositivo responderá 
“si” o “no”; esta respuesta puede cifrarse por medio de los bits 
de estado o de alguna otra forma (véase figura 6.9). 


Figura 6.9 
Acoplamiento (entrada). 


REGISTRO 
DE 
ENTRADA 


DISPOSITIVO 
DE 
ENTRADA 


REGISTRO 
DE 


ESTADO 


El acoplamiento se denomina también “apretón de manos”, 
por analogía con la forma habitual impuesta por la cortesía de 
intercambiar información con una persona con la que no se 
tiene relación frecuente: antes de preguntarle nada, se le saluda 
y se le tiende la mano. El procedimiento de comunicación con 
dispositivos de entrada/salida sigue un ritual parecido, que 
ilustraremos a continuación con un ejemplo sencillo, 


ENVIO DE UN CARACTER A LA IMPRESORA 


Supondremos que el carácter se encuentra en la posición de 
memoria CAR. El programa de impresión es el siguiente: 


ESPERA ІМ — A,(ESTADO) 
BIT 7А COMPROBAR SI 

ESTA LISTA 

EN CASO CON- 


TRARIO, ESPE- 


JR Z, ESPERA 


RAR 

LD А, (CAR) TOMAR EL CA- 
RACTER 

OUT (РКІМТЕ),А IMPRIMIRLO 

JR ESPERA IR A POR EL SI- 
GUIENTE 


El programa es bastante fácil de comprender y utiliza el 
procedimiento de contacto expuesto en la sección anterior. La 
figura 6.10 recoge las trayectorias de los datos. 

El carácter (llamado DATO) se encuentra en la posición de 
memoria CAR. En primer lugar, se comprueba el estado de la 
impresora. Si el bit 7 del registro de estado se convierte en 1, es 
que el dispositivo está listo para aceptar la entrada, es decir, 
que su memoria auxiliar está disponible. En ese momento se 
carga el carácter en el acumulador, desde donde se envía a la 
salida. Mientras el bit de estado sea 0, el programa continuará 
describiendo el bucle del programa llamado ESPERA. 
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ESTADO 


IMPRIMIR 


IMPRESORA 


Figura 6.10 
Trayectoria de los datos hacia 
la impresora. MEMORIA 780 


Ejercicio 6.14: ¿Cuántas instrucciones podrían ahorrarse en el 
programa anterior cargando el dato directamente en el regis- 
tro С y llevándolo también directamente a la salida desde el 
mismo? 


Ejercicio 6.15: Cuando se utiliza una impresora, casi siempre es 
necesario enviar una orden de arranque antes de usarla. 
Modifíquese el programa propuesto para generar dicha orden, 
suponiendo que se obtiene escribiendo un 1 en el bit 0 del 
registro ESTADO, que es bidireccional. 


Ejercicio 6.16: Si no existiese la instrucción BIT, ¿podría sus- 
tituirse por otra en la línea 2 del programa? En caso afirmativo, 
explíquese la ventaja de utilizar BIT, si es que tiene alguna. 


Ejercicio 6.17: Modifiquese el programa propuesto en el sentido 
de imprimir una serie de n caracteres, siendo n menor que 255. 


Ejercicio 6.18: Modifiquese el programa propuesto en el sentido 
de imprimir una serie de caracteres que debe interrumpirse 
cuando se encuentre un código de "retorno de carro". 


Vamos ahora a complicar el procedimiento de salida exi- 
giendo un código de conversión y dirigiendo los datos simultá- 
neamente a varios dispositivos. 


SALIDA A UNA PANTALLA LED DE SIETE 
SEGMENTOS 
Estas pantallas de diodos luminosos (LED) presentan las 


cifras “0” a “9” 6 “0” a “F” hexadecimales iluminando diferentes 
combinaciones de un conjunto de 7 segmentos, como el que 


450 


Figura 6.11 
LED de siete segmentos. 


Figura 6.12 

Caracteres hexadecimales re- 
presentados con LED de siete 
segmentos. 


aparece en la figura 6.11. La 6.12 recoge los caracteres que 
pueden generarse en una pantalla de este tipo. 


Los segmentos del LED se identifican mediante las letras 
“a” a “g” (véase figura 6.11). Así, para representar “0” se 
iluminan los segmentos abcdef. Supongamos ahora que el bit 0 
de una puerta se conecta al segmento “a”; el 1, al “b”; etc.; el 
bit 7 no se utiliza. En estas condiciones, el código binario 
necesario para iluminar fedcba (para representar el valor “0”) 
será “0111111”, que en hexadecimal equivale a “ЗЕ”. Realice el 
ejercicio que se propone a continuación. 
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Ejercicio 6.19: Calcule los equivalentes en siete segmentos de las 
cifras hexadecimales “0” a “F”, y anótelos en la tabla de 
abajo. 


Cód. LED Cód. ES Cód. LL Cód. LED 


| 
6 
7 


Pasemos ahora a la representación de valores hexadecimales 
en varios LED. 


CONTROL DE VARIOS LED 


Un LED no tiene memoria, y representa datos en tanto en 
cuanto sus segmentos estén activados por la corriente. Para que 
la pantalla de diodos resulte barata, el microprocesador repre- 
senta la información secuencialmente, primero en un LED, luego 
en el siguiente, etc.; en evitación de parpadeo, la rotación de 
unos a otros debe ser rápida, inferior a 100 milisegundos. 
Vamos, pues, a diseñar un programa que satisfaga esta condi- 
ción. Utilizaremos el registro С para señalar el punto del LED 
en que deseamos representar una cifra, cuyo valor hexadecimal 
estará contenido en el acumulador. El primer paso será conver- 
tir este valor hexadecimal en su representación equivalente en 
siete segmentos. En el último ejercicio hemos creado una tabla 
de equivalencias, a la que ahora accederemos con direcciona- 
miento indexado, de manera que sea el propio valor hexadeci- 
mal el que proporcione el índice de desplazamiento. Asi, el 
código de siete segmentos correspondiente a la cifra hexadecimal 
“3” se encuentra en el tercer elemento de la tabla a partir de la 
base; llamaremos a la dirección de ésta SEGBAS. El programa 
es: 


LEDS LD E,A A СОМТІЕ- 
NE UNA CI- 
FRA HEX 
LD D.0 UTILIZA 
"DE" COMO 
DESPLAZA- 
MIENTO 


RETARDO 


OUT 


LD 


ADD 


LD 


LD 


OUT 


DEC 


JR 
LD 


DEC 
CP 


JR 
LD 


RET 


HL, SEGBAS 


HL, DE 


A, (HL) 


B, 50H 


(C), A 


NZ, RETARDO 
A,C 


С 
MINLED 


NZ, OUT 
BC, (MAXLED) 


UTILIZA 
“HL” COMO 
INDICE 
DIRECCION . 
DE LA TA- 
BLA 

LEE UN СО- 
DIGO EN 
LA TABLA 
VALOR DE 
RETRASO= 
CUALQUIER 
NUMERO 
GRANDE 
SALIDA DU- 
RANTE EL 
TIEMPO FI- 
JADO 
CONTADOR 
DE RETRA- 
SO 

SIGUE EN 
EL BUCLE 
C ES EL NU- 
MERO DE 
PUERTA 


¿ESTA HE- 
CHO EL UL- 
TIMO LED? 


EN | CASO 
AFIRMATI- 
VO, REINI- 
CIAR C AL 
PRIMERO 


El programa supone que el registro C contiene la dirección 
del LED que debe iluminarse a continuación, y que el acumula- 


dor A alberga la cifra que va a representarse. 


Primero busca el código de siete segmentos correspondiente 
al valor hexadecimal contenido en el acumulador. Como campo 
de desplazamiento se usan los registros D y E; los Н у L sirven 


453 


como registro de índice de 16 bits. La cifra hexadecimal se 
suma a la dirección de la base de la tabla: 


LEDS LD E, А CODIGO DE 7 
SEGMENTOS 
LD D, 0 
LD HL, SEGBAS 
ADD HL,DE 


A continuación se recorre un bucle de retardo para que el 
código obtenido en la tabla se represente durante un espacio de 
tiempo adecuado; en este caso se ha escogido arbitrariamente la 
constante hexadecimal “50”: 


LD A,(HL) LEE EL CODIGO EN LA TABLA 
LD В, 50Н VALOR DE RETARDO 


El retardo se obtiene mediante un bucle. La primera instruc- 
ción: 


RETARDO OUT (С), А 


saca el contenido del acumulador a la puerta de E/S señalada 
por el registro C (el número LED). Las siguientes dos instruc- 
ciones ejecutan el bucle de retardo: 


DEC В 
JR NZ, RETARDO 


Al término del retardo no hay más que decrementar el punte- 
го LED y adoptar las medidas necesarias para ir hasta la 
dirección LED más alta, si ya se ha pasado por la más baja: 


LD A,C 

DEC С 

СР MINLED 

JR NZ, OUT 

LD BC, (MAXLED) 
OUT RET 


Se supone que el programa es una subrutina, y por eso 
acaba con la instrucción RET (retorno de subrutina). 


Ejercicio 6.20: Por lo general, antes de representar una cifra es 
preciso desconectar los amplificadores de los LED. Modifíque- 
se el programa anterior en consecuencia, añadiendo las ins- 


trucciones necesarias (antes de llevar el carácter a la salida se 
lleva el código de carácter “00”). 


Ejercicio 6.21: ¿Qué ocurriría en la pantalla si se llevase retardo 
una línea más arriba? ;Modificaria esto la duración? ¿Influiria 
en la representación de la pantalla? 


Ejercicio 6.22: Como habrá observado, las primeras cuatro ins- 
trucciones del programa ejecutan, en realidad, un acceso 
indexado a la memoria de 16 bits que, debido a no utilizar el 
mecanismo de indexación, parece un tanto confuso. Suponga- 
mos que se conoce de antemano la dirección SEGBAS; llame- 
mos SEGBSS a la parte superior de la misma y SEGBSI a la 
inferior, y almacenemos SEGBSS en la mitad superior del 
registro IX. Vuelva a escribir el programa propuesto utilizan- 
do el mecanismo de direccionamiento indexado del Z80 y 
SEGBSI como campo de desplazamiento de la instrucción. 
¿Qué ventajas y qué inconvenientes derivan de este enfoque? 


Ejercicio 6.23: El programa anterior es una subrutina, y, como 
habrá observado, utiliza internamente los registros B, D, E, H 
y L. Si la subrutina puede usar libremente las áreas de 
memoria designadas por las direcciones T1, T2, T3, T4 y T5, 
¿sería capaz de añadir al principio y al final del programa las 
instrucciones necesarias para garantizar que, cuando la subru- 
tina retorne, el contenido de los registros В, D, E, H y Lsea el 
mismo que cuando entró? 


Ejercicio 6.24: Repita el mismo ejercicio de antes, pero suponiendo 
que las áreas ТІ, T2, ек., no están disponibles para la 
subrutina. ( Un consejo: recuerde que todos los ordenadores 
incorporan un mecanismo que conserva la información en 
orden cronológico.) 


Ya hemos resuelto varios problemas habituales de entrada/ 
salida. Vamos a considerar el caso de un periférico típico: el 
teletipo. 


ENTRADA/SALIDA POR TELETIPO 


El teletipo es un dispositivo en serie que envía y recibe 
palabras de información en formato serie. Cada carácter se 
codifica en formato ASCII de 8 bits (la tabla ASCII se encuenra 
al final de este libro) y, además, va precedido de un bit de 
"arranque" y acabado por dos de "fin". En la conexión llamada 
de bucle de 20 miliamperios, que es la más común, el estado de 
la línea es normalmente “1”, lo que indica al procesador que 


455 


Figura 6.13 
Formato de palabra en el tele- 
tipo. 
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dicha línea no se ha cortado. El arranque es una transición de 
“1” a “0”, y señala al dispositivo receptor que a continuación 
empezarán a llegar bits de datos. El teletipo normal es un 
aparato de 10 caracteres por segundo. Acabamos de subrayar 
que cada carácter exige 11 bits, lo que significa que el teletipo 
debe transmitir 110 bits por segundo o bien 110 baudios. 
Vamos ahora a diseñar un programa que envíe datos en serie a 
un teletipo a la velocidad correcta. 


IMPULSO DE ARRANQUE 2 IMPULSOS DE PARADA 


Ca. 
PARADA 1 ¡PARADA 2 


MARCA - --— 
ESPACIO -- — —— 


9.09 Ms —> 
І 


Si se envían 110 bits por segundo, hay una separación entre 
bit y bit de 9,09 milisegundos, que debe ser la duración del 
bucle de retardo introducido entre bits sucesivos. El formato de 
la palabra en el teletipo aparece en la figura 6.13 y el diagrama 
de flujo de la entrada de bits en la 6.14. El programa es el 
siguiente: 


TTIN IN A, ESTADO 

BIT ТА ¿DATOS LIS- 
TOS? 

JR Z, TTIN ESPERAR, EN 
CASO CONTRA- 
RIO 

CALL RETARDOI CENTRO DEL 
IMPULSO 

IN A,(TTBIT) BIT DE ARRAN- 
QUE 

OUT (ТТВІТ), А DEVOLVERLO 

CALL  RETARDO9 IMPULSO  SI- 
GUIENTE (9MS) 

LD B, 08H CONTADOR DE 
BITS 

BUCLE IN А, (TTBIT) LEER BIT DE 

DATO 

OUT (ТТВІТ,А DEVOLVERLO 

SRL А GUARDARLO 
EN EL АСА- 
RREO 


Figura 6.14 
Entrada a teletipo con devolu- 
ción. 


TTYIN 


¿BIT 
DE ARRANQUE? 


SI 


ESPERAR 4.5 ms 
DEVOLVER BIT 
DE ARRANQUE 


ESPERAR 9.09 ms 


DESPLAZAR BIT 
DE DATO 
DEVOLVERLO 


¿CARACTER 
COMPLETO? 


SI 


ESPERAR 9.09 ms 


BIT DE PARADA 
A LA SALIDA 


ESPERAR 13.59 ms 
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RR C CONSERVARLO 
EN C 
CALL КЕТАКОО9 IMPULSO Sli- 
GUIENTE (9MS) 
DEC B DECREMENTAR 
EL CONTADOR 
DE BITS 
JR NZ, BUCLE 
IN А, (ТТВІТ) ГЕЕВ ВІТ DE 
PARADA 
OUT  (TTBIT) А DEVOLVERLO 
CALL  RETARDO9 SALTAR AL SF- | 
Figura 615 GUNDO ALTO 
Programa de teletipo. RET 
Vamos a examinar el programa con cierto detalle. Lo 
primero es verificar el estado del teletipo para determinar si el 
carácter está disponible: 
тим ІМ A, ESTADO 
BIT ТА 
JR Z, TTIN 
La instrucción BIT es un recurso útil del 280 que permite 
verificar cualquier bit de cualquier registro de datos sin modifi- 
car su contenido. La bandera Z se activa si el bit vale 0, y se 
pone a O en caso contrario. | 
El programa, por tanto, recorrerá un bucle hasta que el 
estado pase a “1”; es un bucle de barrido clásico. 
Obsérvese que, como no es preciso mantener el contenido de | 
ESTADO, también podría haberse usado la instrucción | 
AND 10000000В 
en lugar de 
| 
BIT 7,А 
aunque a costa de destruir el contenido de A (cosa que, en este 
caso, sería aceptable). 
Al optimizar un programa, hay que tener en cuenta que 
cada instrucción nueva puede producir efectos secundarios. 
A continuación se ejecuta un retardo de 4.5 ms para detec- 
tar el bit de arranque en el centro del impulso: 
CALL RETARDO1 | 
| 
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Figura 6.16 
Entrada al teletipo. 


RETARDO]! es la subrutina de retardo encargada de producir- 
lo. El primer bit que llega es el de arranque, que debe devolver- 
se al teletipo e ignorarse; de ello se encargan las instrucciones 
siguientes: 


ТТМ IN А, (ТТВІТ) 
OUT (ТТВІТ), А 


A continuación hay que esperar la llegada del primer bit de 
dato; el retardo necesario es de 9.09 milisegundos, y lo produce 
una subrutina: 


CALL КЕТАКОО9 


El registro B se usa como contador, y se carga con el valor 
8 para captar datos de 8 bits: 


LD В,08Н 


А continuación los bits se leen uno por uno еп el acumula- : 
dor y se devuelven. Se supone que llegan en la posición 0 al 
acumulador, tras lo que se guardan en el registro C, donde se 
desplazan. La transferencia de А a С se lleva a cabo mediante el - 
bit de acarreo: 


BUCLE IN А, (TT BIT) 
OUT (ТТВІТ), А 
SRL А 
КК С 
Га secuencia se ilustra еп la figura 6.16. 


ESPACIO DE E/S 


m TELETIPO 
CONTADOR 


A continuación se produce el retardo habitual de 9 milise- 
gundos, se decrementa el contador de bits y se repite el bucle, si 
todavía no se han recibido los 8 bits: 
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mme 


CALL КЕТАКрО9 
DEC B 
JR NZ, BUCLE 


Por ültimo, se recoge y se devuelve el bit de parada. Por lo 
general, basta con uno, pero pueden enviarse dos con un par de 
instrucciones adicionales: 


IN А, (TTBIT) 
OUT  (TTBIT) А 
CALL  RETARDO9 
RET 


El programa debe estudiarse con atención. La lógica es 
bastante sencilla, y lo único nuevo es que todos los bits se leen 
en el teletipo (en la dirección TTBIT) y se devuelven al mismo. 
Esto es una característica propia de todos estos aparatos; 
cuando el usuario pulsa una tecla, la información se transmite 
al procesador, que la devuelve acto seguido a la impresora del 
teletipo. De esta forma se comprueba que las líneas de transmi- | 
sión están en funcionamiento y que el procesador actúa correc- | 
tamente. 


ENTRADA 


ENTRADA 


ENVIAR BIT are 
DE ARRANQUE pes | 


ENVIAR BITS 
DE DATOS 
ENVIAR BIT 
DE PARADA 


SALIDA 


UN BIT 
A LA SALIDA 


RETRASO 
DE 9.1 MSEG 


TERMINADO 


Y Si 


Figura 6.17 
Salida del teletipo. 


Ejercicio 6.25: Escriba la rutina necesaria para producir un 
retraso de 9.09 milisegundos (subrutina RETARDOS). 


Ejercicio 6.26: Utilizando como ejemplo el programa que acaba- 
mos de examinar, escriba otro PRINTC que imprima en el 
teletipo el contenido de la posición de memoria CAR (véase 
figura 6.16). 


He aquí la solución: 


PRINTC LD B, 11D CONTADOR 

= 11 BITS 

LD А, (CAR) TOMAR EL 
CARACTER 

OR A BORRAR EL 
ACARREO 
= BIT DE 
ARRANQUE 

RLA ACARREO EN 
A 

BUCLE OUT  (TTBIT,A SALIDA 

CALL RETARDO 

RRA BIT SIGUIEN- 
TE 

SCF ACARREO = 1 
(BIT DE PARA- 
DA) 

DEC В CONTADOR 
DE BITS 

JR NZ, BUCLE 

RET 


El registro B se utiliza como contador de bits para la 
transmisión. El contenido del bit 0 de A se envía a la línea del 
teletipo (“TT BIT”). Obsérvese de qué forma se utiliza el acarreo 
para proporcionar el noveno bit (bit de ARRANQUE). Nótese 
también que el acarreo se borra mediante 


OR A 
Al final del programa, el acarreo se activa a 1 mediante 
SCF | 


para generar un bit de parada. 


Ejercicio 6.27: Modifique el programa para que espere un bit de 
ARRANQUE en lugar de un bit de ESTADO. 
IMPRESION DE UNA SERIE DE CARACTERES 


Supondremos que la rutina PRINTC (véase ejercicio 6.26) se 
encarga de imprimir un carácter en la impresora o de presentar- 
lo en cualquier dispositivo de salida. Vamos ahora a imprimir el 
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Figura 6.18 
Impresión de un bloque de me- 
moria. 


contenido situado entre las posiciones de memoria (PRIMER y 
(PRIMER + N). El programa es bastante simple (véase figura 
6.18): 


PSERIE LD B, NBR LONGITUD 
DE LA SERIE 
LD HL, PRIMER DIRECCION ` 


DE BASE 
BUCLE LD А, (HL) TOMAR UN 
CARACTER 
CALL  PRINTC IMPRIMIRLO 
INC HL ELEMENTO 
SIGUIENTE 
DEC B 
JR NZ, BUCLE REPETIR 
RET 


MEMORIA 


B 


CONTADOR 


ARRANQUE + М 


A LA IMPRESORA 


REGISTRO DE SALIDA 


Resumen de periféricos 


Ya hemos descrito las técnicas de programación básicas 
para comunicarse con dispositivos de entrada/salida típicos. 
Pero, además de transferir datos, es preciso preparar uno о más 
registros de control dentro de cada uno de los dispositivos de 
E/S para organizar las velocidades de transferencia, el mecanis- 
mo de interrupciones y otras varias posibilidades. A tal efecto es 
preciso consultar el manual que acompaña al dispositivo de que 
se trate (para más detalles sobre los algoritmos especificos de 
intercambio de información con todos los periféricos habituales, 
se remite al lector a la publicación Microprocessor Interfacing 
Techniques, de SYBEX, EE.UU., 2344 Sixth Street, Berkeley, 
California 94710. 


Sabemos manejar dispositivos individuales, pero en un siste- 
ma real todos están conectados a los buses y pueden solicitar 
atención simultáneamente. ¿Cómo se organiza el tiempo del 
procesador? 


Organización de la entrada/salida 


Como las solicitudes de entrada/salida pueden presentarse 
simultáneamente, es preciso crear un esquema de organización 
para determinar el orden en que se atenderán. Para ello se 
emplean tres técnicas básicas que pueden combinarse entre sí: 
muestreo de estaciones, interrupción, DMA. Describiremos aquí 
el muestreo y la interrupción, pero no el ОМА, que es un 
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dispositivo del soporte físico. 


MUESTREO DE ESTACIONES 


Desde el punto de vista conceptual, constituye la forma más 
fácil de manejar varios periféricos. El procesador interroga uno 
por uno a todos los dispositivos conectados a los buses; si uno 
de ellos solicita servicio, se lo proporciona; en caso contrario, se 
pasa a interrogar al periférico siguiente. La técnica es aplicable 
a cualquier rutina de servicio de un dispositivo. 

Por ejemplo: si el sistema dispone de un teletipo, una 
grabadora de cinta magnética y una pantalla de rayos catódi- 
cos, la rutina de barrido preguntaría al teletipo: “¿tienes algún 
carácter que transmitir?”; a continuación se dirigiría a la rutina 
de salida del teletipo: “¿tienes que enviar algún carácter?” Si las 
dos respuestas fuesen negativas, interrogaría a las rutinas del 
aparato de cinta magnética y, por último, a la pantalla. Cuando 
sólo hay un dispositivo conectado al sistema, también puede 
usarse el muestreo para determinar si necesita servicio. Como 
ejemplo, las figuras 6.21 y 6.22 recogen los diagramas de flujo 
de lectura de una máquina lectora de cinta de papel y de 
impresión en una impresora. 

El programa propuesto a continuación ejecuta un bucle de 
muestreo de los dispositivos 1, 2, 3 y 4 (véase figura 6.20): 


MUESTREO4 IN А, (ESTADO1) ESTADO DEL 
DISPOSITIVO 
1 
BIT TA ` ¿SOLICITUD 


DE SERVICIO? 
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CALL М2, UNO ¿BIT 7= 1? 

IN A,(ESTADO2) DISPOSITIVO 
2 

BIT 7, А 

CALL NZ, DOS 

IN A,(ESTADO3) DISPOSITIVO 
3 

BIT 7, А 


CALL М7,ТВЕ5 
IN A,(ESTADO4) DISPOSITIVO 
4 


BIT 7,A 

CALL М7, CUATRO 

JR MUESTREO4 NO HAY RES- 
PUESTA, 
PROBAR DE 
NUEVO 


El bit 7 del registro de estado de cada dispositivo es “1”, si 
desea servicio. Cuando detecta la solicitud, el programa salta al 
controlador del dispositivo, situado en la dirección UNO, para 
el 1; en la DOS, para el 2, etc. 


BUS DE DATOS 
MUESTREO 


INTERRUPCION 


pco [ue] (Com 
E : rum 
ғ 

| шиш 


ОМА 


Figura 6.19 
Tres métodos de control de 
$. 


Hay un punto que conviene subrayar; es importante verifi- 
car de qué forma afectan a los códigos de condición cada una 
de las instrucciones. IN A no modifica las banderas, pero si еп 


Figura 6.20 
Diagrama de flujo del bucle del 
muestreo. 


Figura 6.21 
Lectura de una máquina lectora 
de cinta de papel. 


su lugar se hubiese utilizado INr, el bit 7 de la entrada 
reflejaría automáticamente el bit del SIGNO del registro de 
estado, y la instrucción BIT 7, А sería innecesaria; pero como 
IN A no cambia las banderas, es necesario incluir en el progra- 
ma esa comprobación adicional. 

En algunos soportes físicos, los dispositivos de entrada/sali- 
da pueden tratarse, a efectos de direccionamiento, como posi- 
ciones de memoria (entrada/salida por zona de memoria). En 
tal caso, la instrucción IN podría sustituirse por otra LD, 
dejando igual el resto del programa, porque LD no afecta a las 
banderas. 


A 


¿PIDIENDO 
SERVICIO? 


RUTINA DE SERVICIO 
PARA EL DISPOSITIVO A 


B 


¿PIDIENDO 
SERVICIO? 


RUTINA DE SERVICIO 
PARA EL DISPOSITIVO B 


с 


¿PIDIENDO 
SERVICIO? 


RUTINA DE SERVICIO 
PARA EL DISPOSITIVO С 


ACTIVAR 
LA LECTORA 


LEER CARACTER 
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Figura 6.22 
Impresión con una perforadora 
o una impresora. 


Las ventajas del muestreo son obvias: es una técnica senci- 
lla, no necesita apoyo del soporte físico y mantiene todas las 
entradas y salidas sincronizadas con el programa. El inconve- 
niente también salta a la vista: la mayor parte del tiempo del 
procesador se pierde interrogando a dispositivos que no necesi- 
tan atención, lo que, entre otras cosas, podría hacer que llegase 
tarde a prestarlo a los que sí lo necesitan. 

Por tanto, conviene disponer de otro mecanismo que deje al 
procesador libre para realizar cálculos útiles, en lugar de obli- 
garle a interrogar continuamente a los periféricos. De todas 
formas, hay que insistir en que el muestreo se emplea muchisi- 
mo cuando el microprocesador no está muy sobrecargado, 
porque facilita mucho la organización del sistema. Estudiemos 
ahora la alternativa más importante al muestreo: la interrup- 
ción. 


CARGAR EL BUFFER 


DE LA PERFORADORA 
O IMPRESORA 


TRANSMITIR 
EL DATO 


INTERRUPCIONES 


La idea de interrupción puede verse en la figura 6.19. Una 
línea especial del soporte físico, llamada línea de interrupciones, 
se conecta a una patilla específica del microprocesador. А esa 
línea pueden conectarse numerosos dispositivos de entrada/sali- 
da, y sirve para comunicar por medio de una señal que uno de 
ellos solicita atención al procesador. Veamos de qué forma 
responde éste a la solicitud. 

En cualquier caso, el procesador termina de ejecutar la 
instrucción en curso, porque de otra forma se produciría un 
caos en su interior. Á continuación salta a una rutina de 


Figura 6.23 
La pila del Z80 tras una inte- 
rrupción. 


Figura 6.24 
Protección de algunos registros 
de trabajo. 


manipulación de interrupciones, que se encargará de llevar el 
caso; como se produce un salto, es necesario guardar en la pila 
el contenido del contador del programa; por tanto, cualquier 
interrupción debe determinar automáticamente la conservación del 
contador del programa en la pila. También hay que conservar 
automáticamente el registro de estado F, ya que su contenido se 
verá alterado por las instrucciones que sigan. Y, por último, será 
preciso preservar en la pila todos los registros internos suscepti- 
bles de ser modificados por la rutina (véanse figuras 6.23 y 
6.24). 


SP PCL 


PCH 


Una vez a salvo todos los registros que sean menester, 
puede saltarse a la dirección de manipulación de interrupciones. 
Al final de la rutina se restauran todos los registros y se ejecuta 
un retorno de interrupción especial, que tiene por objeto permi- 
tir la continuación del programa principal. Vamos a estudiar 
con más detalle la línea de interrupciones del Z80. 


INTERRUPCIONES EN EL Z80 


Una interrupción es una señal no sincronizada con el 
programa enviada al microprocesador en cualquier momento en 
solicitud de servicio. Cuando un programa salta a una subruti- 
na, se dice que ésta está sincronizada con aquél, es decir, 
organizada por él; por el contrario, la interrupción puede 
producirse en cualquier momento y, por lo general, suspender 
la ejecución del programa en curso sin que éste lo "sepa". 
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Como puede producirse en cualquir momento, se dice que es 
asíncrona. 

El Z80 dispone de tres mecanismos de interrupción: solici- 
tud del bus (BUSRQ), interrupción no enmascarable (NMI) e 
interrupción normal (INT) A continuación estudiaremos los 
tres. 


SOLICITUD DEL BUS 


Es el mecanismo de interrupciones de máxima prioridad del 
780. La secuencia que desencadena aparece en la figura 6.25, 
Como norma general, el Z80 no percibirá ninguna interrupción 
hasta que no acabe con el ciclo de máquina en curso. Las | 
interrupciones NMI e INT no se obedecen hasta que no 
termina la instrucción que se está ejecutando, pero BUSRQ se 
atiende en cuanto acaba el ciclo en curso, sin esperar a que 
termine la instrucción completa. Se utiliza para el acceso directo | 
a la memoria (DMA), y hace que el 780 pase a este modo. Si se 
ha llegado al término de una instrucción y hay pendientes 
señales NMI o INT, el 780 las memoriza internamente, activan- 
do los correspondientes biestables especiales. En modo DMA, el ) 
procesador suspende todas las operacions y pasa los buses de 
datos y direcciones al estado de alta impedancia; este modo es 
utilizado habitualmente por un controlador DMA para efectuar | 
transferencias entre un dispositivo de entrada/salida de gran | 
velocidad y la memoria, empleando para ello los buses de datos i 
y direcciones. El término de la operación se indica mediante 
una serie de niveles cambiantes BUSRQ, momento en que el 
Z80 reanuda el funcionamiento normal, empezando por com- 
probar si están activados los biestables NMI o INT para | 
atender las interrupciones correspondientes. 

Salvo que la sincronización sea muy importante, el progra- 
mador no tendrá que preocuparse por el DMA. Lo único que 
deberá tener en cuenta es que, si el sistema cuenta con un 
controlador de DMA, este modo puede retrasar la respuesta a ( 
las interrupciones ММГ e INT. 


Interrupción no enmascarable 


Debe su nombre a que el programador no puede impedirla. 
El 780 la acepta siempre al término de la instrucción en curso, 
salvo que haya recibido solicitud por bus; si ММГ se recibe 
durante una operación BUSRQ, se activa el biestable interno 
NMI y se atiende al término de la instrucción que sigue a 
BUSRQ. 


Figura 6.25 
Secuencia de una interrupción. 


м 
NO 
ENMASCARABLE 


DE 
INTERRUPCION 
ENMASCARABLE 


ММГ provoca el PUSH automático del contador del pro- 
grama en la pila y el salto a la dirección 0066H, cuyos dos 
bytes pasan al contador mencionado; representan, pues, la 
dirección de partida de la rutina de manipulación de NMI 
(véase figura 6.26). 

Este mecanismo de interrupción se utiliza en situaciones de 
“emergencia”, y no ofrece la flexibilidad de la interrupción 
enmascarable que explicaremos más adelante. 

Téngase en cuenta que hay que cargar una rutina de inter- 
rrupciones en la dirección 0066H antes de usar NMI. 

Cuando entra en acción, NMI desencadena la siguiente 
secuencia de operaciones: 


SP .— SP – 1 


(SP) <— РСН 
ан BUSH де Be 


(SP) + PCL 


E 


Figura 6.26 
ММ! provoca la vectorización 
automática. 
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MEMORIA 


MANIPU- 
LADOR 
DE NMI 


pila 


Acto seguido, NMI provoca la reanudación automática en 
la posición 066H. La secuencia completa de acontecimientos es 
la siguiente: 


РС ——Ф PILA (salvaguarda el contador del 
programa) 
IFFI —— IFF2 (salvaguarda IFF) 
0 ТЕГІ (reinicia 1FF) 
SALTO A 0066H (activa el manipulador de in- 
terrupciones) 


El estado del biestable de interrupciones filtrables IFF1 
presente en el momento de recepción de NMI se conserva 
automáticamente en IFF2, y a continuación se reinicia para 
evitar ulteriores interrupciones. Esto es importante para impedir 
la pérdida de la prioridad inferior de INT y simplfica el soporte 


“físico externo, porque el estado de cualquier INT pendiente se 


conserva internamente en el Z80. 

La interrupción NMI suele utilizarse en situaciones de alta 
prioridad, como reloj de tiempo real o fallo de la alimentación. 

El retorno de ММГ se encarga a una instrucción especial: 
RETN, retorno de una interrupción no enmascarable, en virtud 
de la cual se restaura el contenido de ІЕЕІ, a partir de IFF2, y 
el del contador del programa, a partir de su posición en la pila. 
Como IFF1 se habia reiniciado durante la ejecución de NMI, 
durante ésta no puede haberse aceptado ninguna INT externa 
(salvo que el programador haya incluido una instrucción El 
dentro de la rutina NMI); no ha habido, pues, pérdida de 
información. 


Cuando acaba el manipulador de interrupciones, se produce 
la siguiente secuencia: 


IFF2 ——— IFFI (restaura el ПЕЕ) 
PILA — РС (restaura el contador del programa) 


Obsérvese que con el contenido de IFF1 se restaura la 
situación de admisión de interrupciones enmascarables. 


Interrupción | 


La interrupción normal enmascarable INT puede actuar en 
tres modos, específicos del Z80, porque el 8080 sólo dispone de 
uno. El programador puede enmascarar selectivamente la inte- 
rrupción normal INT: cuando los biestables IFF1 e IFF2 están 
en “1”, el procesador recibe interrupciones; cuando se enmasca- 
ran a “0”, las ignora. Estos biestables se activan a “1” mediante 
la instrucción El, y a “0” mediante DI (los dos se activan y se 
reinician simultáneamente). Para evitar pérdidas de información, 
las interrupciones INT se ignoran durante la ejecución de las El 
y DI. Pasemos ya a estudiar los tres modos de interrupción. 


Modo de interrupción 0 


Este modo es idéntico al de 8080. El Z80 opera en modo 0 
cuando se pone en marcha inicialmente (cuando se aplica la 
señal RESET) o cuando se ejecuta una instrucción ГМО. Una 
vez en este modo, se detecta una interrupción si el biestable 
IFF1 se encuentra en 1, siempre que no se produzca al mismo 
tiempo una solicitud del bus o una interrupción no enmascara- 
ble. La interrupción no se detecta hasta el término de la 
instrucción en curso. El Z80 responde a la misma generando 
una señal ТОКО y otra МІ, pasando a situación de espera sin 
hacer nada más. 

Las señales IORQ y MI debe detectarlas un dispositivo 
externo en una operación llamada identificación de interrupción 
o INTACK, y a continuación depositará una instrucción en el 
bus de datos. En el ciclo siguiente, el Z80 espera a que el 
dispositivo externo deposite en el bus dicha instrucción, que 
habitualmente es RST o CALL. Las dos guardan automática- 
mente el contador del programa en la pila y provocan el salto a 
una dirección determinada. La ventaja de RST es que ocupa un 
solo byte; por tanto, se ejecuta rápidamente; tiene el inconve- 
niente de que sólo puede saltar a una de ocho posiciones de la 
página O (direcciones 0 a 255). La instrucción CALL tiene la 
ventaja de ser un salto de tipo general que especifica una 
dirección completa de 16 bits, aunque, como exige tres bytes, se 

ejecuta con más lentitud. 
“| 
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Obsérvese que se ignoran todas las interrupciones que 
pudieran llegar a partir del instante en que empieza a tratarse 
una, porque IFF1 e IFF2 pasan automáticamente а “0”. A 
partir de este punto, el programador debe insertar una instruc- 
ción El (que admite interrupciones) en la posición adecuada 
—y, en cualquier caso, antes del retorno de la interrupción en 
curso— si desea admitir otras adicionales. 


La figura 6.27 recoge la secuencia que corresponde a una 
interrupción de modo 0. 


MODO O MODO 1 MODO 2 


IGNORAR 
INTERRUPCIONES 


ДЕЕ EL PRIMER BYTE 
DE LA INSTRUCCION 


IGNORAR 
INTERRUPCIONES 


РС» PILA 


IGNORAR 
INTERRUPCIONES 


n 


LEER VECTOR 


(M1, ЮАО. BAJO) 


SALYAR A 0038H 
. 


fORMAR VECTOR 
TABLA DIRECCIONES 
REG 1 + VECTOR 


. 
. 
El (ADMITIR 
INTERRUPCIONES) 


¿HACEN FALTA 
MAS BYTES PARA 
LA INSTRUCCION? 


NO 


DIRECCION 
DE PARTIDA 


LEER BYTE SIGUIENTE DE LA TABLA 


(LECTURA EN MEM. NORMAL 
CON PC ESTACIONARIO) 


-- 


РС» РИА 
но 
EJECUTAR INSTRUCCION 


[SALTAR A NUEVA POSICION] 
INICIAR RUTINA 
SERVICIO INTERRUPCION 


Si 


El (ADMITIR | 
INTERRUPCIONES) | 
RET \ 

PILA > PC 


soto | 
РАВА САЦ. | 
o RST | 


Figura 6.27 
Modos de interrupción. Pre 


El retorno de una interrupción se efectúa mediante una 
instrucción RETI. Recordemos que el programador debe finali- 
zar casi siempre explícitamente la interrupción que ha atendido 
al dispositivo de E/S, y siempre debe restaurar la bandera de 
bloqueo de interrupciones. No obstante, el controlador de 
periféricos puede utilizar la señal INTACK para acabar con la 
solicitud INT y liberar al programador de esa tarea. 

Además, si la rutina de manipulación de interrupciones 
modifica el contenido de cualquiera de los registros internos, es 
responsabilidad exclusiva del programador guardarlos en la pila 
antes de ejecutar dicha rutina. En caso contrario, se destruirían 
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los contenidos de tales registros, y cuando el programa reanu- 
dase el funcionamiento normal se produciría un fallo. Suponga- 
mos, por ejemplo, que el manipulador de interrupciones va a 
utilizar los registros А, В, С, D, E, H y L, el programador 
deberá guardarlos todos en la pila (véase figura 6.28). 


DIRECCIONES 
DESCENDENTES 


Figura 6.28 
Guardar los registros. 


El correspondiente programa es: 


GUARDA PUSH AF 
PUSH BC 


PUSH DE 
PUSH HL 
Al término de la rutina de manipulación de interrupciones 


hay que restaurar todos los registros; el mencionado manipula- 
dor acabará con las instrucciones: 


POP HL 

POP DE 

POP BC 

POP AF 

EI (salvo que El se hubiese utilizado en un pun- 
to anterior de la rutina) 


También deben preservarse y restaurarse los registros IX e 
IY si los utiliza la rutina. 


473 


Figura 6.29 
Modo de interrupción 1. 
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Modo de interrupción 1 


Este modo entra en acción cuando se ejecuta la instrucción 
IM1. Es un manipulador automático de interrupciones que 
provoca el salto a la posición 0038H. Es, pues, una instrucción 
básicamente análoga a la interrupción NMI, con la diferencia 
de que es enmascarable. El 780 guarda automáticamente el 
contenido del PC en la pila (véase figura 6.29). 


INT 0 


vectorización 
automática 


RUTINA 
DE INTERRUPCION 


PROGRAMA 


protección | POSICION 


automática 


0038 
(automático) 


MEMORIA 


Esta respuesta automática que “vectoriza” todas las inte- 
rrupciones a la dirección de memoria 38H tiene su origen en la 
necesidad del 8080 de reducir al mínimo la cantidad de soporte 
físico necesaria para trabajar con interrupciones. Su posible 
inconveniente es que salta sólo a una posición de memoria. Si 
hay varios dispositivos conectados a la línea INT, el programa 
que empiece en la dirección 38H se verá obligado a determinar 
cuál es el dispositivo que solicita atención; más adelante volvere- 
mos sobre este problema. 

Hay que tener cuidado con la sincronización de esta inte- 
rrupción, porque, al efectuar transferencias programadas de en- 
{тада y salida, el 780 ignora todos los datos que pueda haber 
en el bus correspondiene durante el ciclo que sigue a la inte- 
rrupción (ciclo de identificación de la interrupción). 


Modo de interrupción 2 (interrupción vectorial) 


Este modo se activa mediante la instrucción IM2. Es un 
recurso muy potente que permite la vectorización automática de 
interrupciones. El vector de interrupción es una dirección pro- 
porcionada por el dispositivo periférico que genera la parada, y 
que se usa como apuntador de memoria para la dirección de 


DE LA INTERRUPCION 


partida de la rutina de tratamiento de interrupciones. El meca- 
nismo de direccionamiento porporcionado por el Z80 en modo 
2 es indirecto; cada periférico suministra una dirección de salto 
de 7 bits que se añade a la de 8 bits contenida en el registro 
especial del Z80 I; el bit O (situado en el extremo derecho) de la 
dirección de 16 bits se pone a “0”, lo que da lugar a otra 
dirección que apunta hacia la entrada a una tabla situada en 
cualquier lugar de la memoria. La tabla puede albergar hasta 
128 entradas de palabras dobles, cada una de las cuales es la 
dirección del manipulador de interrupciones de un dispositivo. 
Las figuras 6.30 y 6.31 ilustran esta situación. 


2X VECTOR 


DISP VECTOR DE 
OSITIVO j BITS 


| 
Figura 6.30 
Modo de interrupción 2. MEMORIA 


En este modo, el Z80 empuja automáticamente en la pila el 
contenido del contador del programa; es una precaución evi- 
dentemente necesaria, ya que el PC puede cargarse con el 
contenido de la entrada a la tabla de interrupciones correspon- 
diente al vector proporcionado por el dispositivo. 


Servicio de las interrupciones 


En la figura 6.19 se hace una comparación gráfica entre el 
muestreo y la interrupción. Fs fácil ver que la primera técnica 
obliga al programa a perder muchísimo tiempo tomando mues- 
tras. 

El otro mecanismo detiene el programa, es atendido y deja 
que éste siga su curso normal. El inconveniente de la interrup- 
ción es que obliga a añadir varias instrucciones al principio y al 
final que retrasan la ejecución de la primera instrucción del 
manipulador de interrupciones; son los servicios generales de 
este mecanismo. 
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Figura 6.31 
Un ejemplo práctico de mo- 
do 2. 
Ejercicio 6.28: Calcule, a partir de las tablas de ciclos por ins- 
trucción del capítulo 4, el tiempo que se pierde en guardar y 
restaurar los registros A, B, D y H. 


Una vez aclarado el funcionamiento de las líneas de inte- 
rrupción, nos centraremos en dos importantes problemas toda- 
vía pendientes: 

1. ¿Qué hacer cuando varios dispositivos solicitan una in- 

terrupción al mismo tiempo? 

2. ¿Qué hacer si se presenta una interrupción mientras se 

atiende otra? 


VARIOS DISPOSITIVOS CONECTADOS A UNA 
MISMA LINEA DE INTERRUPCIONES 


Cada vez que ocurre una interrupción, el procesador salta a 


una dirección determinada, pero, antes de que pueda hacer 
nada, la rutina de tratamiento de interrupciones debe determi- 
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nar de qué dispositivo procede la señal. Como es habitual, la 
identificación puede hacerse mediante el soporte lógico o me- 
diante el físico. 

En el primer caso se recurre al muestreo; el microprocesa- 
dor pregunta a un dispositivo: “¿has solicitado una interrup- 
ción?” Si la respuesta es negativa, dirige la pregunta al siguiente. 
La figura 6.32 recoge el método. El programa de muestreo es el 
siguiente: 

MUESTRA IN A, (ESTADO) LEE EL ESTA- 

DO 

BIT ТА ¿HA SOLICITA- 
DO INTERRUP- 
CION EL DIS- 


POSITIVO? 

JP | NZUNO EN CASO AFIR- 
MATIVO, ATEN- 
DERLA 

IN  A,(ESTADO?2) 

BIT 7A 

JP | NZ,DOS 

etc. 


El soporte físico proporciona la dirección del dispostivo, a la 
par que la solicitud de interrupción. 


INT 1  MUESTRO INTERRUPCION . VECTORIZADA 


Figura 6.32 
Interrupciones muestreadas у 
vectorizadas. 


RUTINA 
DE SERVICIO 
UTINA 
Para ser más exacto, cuando se opera en modo 0, el contro- 
lador del periférico deposita una instrucción RST de un byte o 


ao a 
DISPOSITIVO? DE MUESTREO 
RUTINA 
R 
una CALL de tres bytes en el bus de datos como respuesta al 
reconocimiento de interrupción, lo que automatiza la vectoriza- 


ción de interrupciones y reduce el tiempo de servicio necesario, 
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Figura 6.33 


Varios dispositivos compartien- 
do una misma línea de inte- 


rrupciones. 
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Téngase en cuenta que hace falta una instrucción de llamada 
a subrutina, porque el Z80 no guarda el PC cuando funciona 
en modo 0. 

En la mayor parte de los casos la velocidad de respuesta a 
una interrupción no es crucial, por lo que se emplea el mecanis- 
mo de muestreo. Pero si ese tiempo es crucial, hay que recurrir 
al soporte físico. 


INTERRUPCIONES SIMULTANEAS 


El siguiente problema es la solicitud de una interrupción 
durante la ejecución de una rutina de manipulación de interrup- 
ciones. Vamos a ver lo que ocurre y cómo puede usarse la pila 
para salir del paso. En el capítulo 2 ya adelantamos que esta 
función era una de las primordiales de la pila, y ahora vamos a 
demostrarlo. El problema general se ilustra en la figura 6.34, en 
la que el tiempo transcurre de izquierda a derecha; la parte 
inferior de la misma recoge el contenido de la pila. Empezando 
por la izquierda, en el momento ТО está en marcha el programa 
P. En ТІ se produce la interrupción 11, que se autoriza. El 
programa P se detiene, como queda indicado en la parte infe- 
rior de la figura. La pila contiene, como minimo, el contador del 
programa y el registro de estado de P, más cualesquiera otros 
registros guardados por el manipulador de interrupciones o por 
la propia 11. 


En Tl empieza a ejecutarse Il. En T2 se produce la inte- 
rrupción 12, que en este ejemplo supondremos de prioridad 
superior a 11. Si hubiera tenido menos prioridad, se habría 
ignorado hasta la terminación de I1. En T2 se llevan a la pila 
los registros de 11, como indica la parte inferior de la ilustra- 
ción; una vez más se guardan en la pila los contenidos del 
contador del programa y de AF. Además, la rutina de I2 podría 
proteger nuevos registros. Á partir de este momento, la inte- 
rrupción 12 se ejecuta hasta que termina en el momento T3. 

Cuando acaba —con una instrucción RETI—, el contenido 
de la pila pasa automáticamente al Z80, como se ve en la parte 
inferior de la figura 6.34, de manera que se reanuda también 
automáticamente la ejecución de I1. Pero en T4 vuelve a produ- 


Figura 6.34 


El contenido de la pila durante 
una serie de interrupciones 


múltiples. 


Resumen 


cirse una nueva interrupción 13 de mayor prioridad; los regis- 
tros de I1 vuelven otra vez a la pila. La nueva interrupción se 
ejecuta entre T4 y T5, punto en el que concluye. El contenido 
de la pila pasa de nuevo al Z80, y prosigue la ejecución de I1, 
que en esta ocasión avanza sin más interrupciones hasta termi- 
nar en T6. En este momento, los registros que estaban guarda- 
dos en la pila pasan al Z80, y continüa la ejecución del progra- 
ma P. Como el lector puede comprobar, cuando esto ocurre, la 
pila ya está vacía (el número de líneas de puntos que señalan la 
interrupción del programa corresponde al de niveles almacena- 
dos en la pila). 


HEMEO т, т, т, т, m. Un 
PROGRAMA P Ai шош жш шоқ асы Om шш с с d 
! 
INTERRUPCION 1, һ------4------- 0—4 = np 
INTERRUPCION 1, 1 Pia 
t 
---+4 


INTERRUPCION 1, 


" по OUR 


Ejercicio 6.29: Supongamos que el área a disposición de la pila en 
un programa determinado está limitado a 300 posiciones, que 
es preciso guardar todos los registros y que el programador 
permite la existencia de interrupciones internas, es decir, de 
interrupciones dentro de interrupciones. ¿Cuál será el máximo 
número de éstas admisible simultáneamente? ¿Hay algún otro 
factor capaz de reducir todavía más ese número? 


Hay que subrayar que los microprocesadores no suelen estar 
conectados a una cantidad grande de dispositivos generadores 
de interrupciones, por lo que la probabilidad de que se produz- 
can muchas simultáneamente es baja. 

Ya hemos resuelto todos los problemas asociados habitual- 
mente a las interrupciones. El mecanismo es fácil de manejar, y 
hasta el programador sin experiencia debe ser capaz de utilizarlo 
con aprovechamiento. 


Hemos pasado revista en este capítulo a todas las técnicas 
de comunicación con el mundo exterior. Desde las rutinas de 
entrada/salida más elementales hasta programas de comunica- 
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ción con periféricos reales, hemos aprendido a desarrollar todas 
las rutinas normales, y hasta hemos analizado la eficacia de un 
par de programas de referencia de transferencia en paralelo y de 
conversión paralelo a serie. La parte final del capítulo se ha 
dedicado a la organización del trabajo con varios periféricos 
mediante los mecanismos de muestreo e interrupción. А un 
sistema pueden conectarse muchos otros dispositivos de entra- 
da/salida de la naturaleza más dispar, pero con las técnicas 
expuestas y si se comprende su funcionamiento, debe ser posible 
resolver la mayor parte de los problemas. 

En el capítulo próximo veremos las características de las 
pastillas de conexión de entrada/salida que suelen acoplarse al 
780, para considerar a continuación las estructuras básicas de 
datos que puede emplear el programador. 


Ejercicio 6.30: Calcúlese la duración del servicio en modo 0 supo- 
niendo que se guardan todos los registros y que se recibe una 
instrucción RST en respuesta a la detección de la interrup- 
ción. El servicio es el retraso total producido cuando se atien- 
de una interrupción, con exclusión del debido a la ejecución de 
las instrucciones necesarias para tratar la interrupción propia- 
mente dicha. 


Ejercicio 6.31: Los LED de 7 segmentos no sólo sirven para 
representar las cifras del sistema hexadecimal. Determínense los 
códigos de representación de: H, Г, J, L, O, Р, 5, U, Y, g, h, i, 
j, l n, o, p ғ, t, u, y. 


Ejercicio 6.32: La figura 6.34 recoge el diagrama de flujo del 
tratamiento de una interrupción. Responda a las siguientes 
preguntas: 


a) ¿Qué hace el soporte lógico y qué el fisico? 

b) ¿Para qué sirve el enmascaramiento? 

c) ¿Cuántos registros deben guardarse? 

d) ¿Cómo se identifica el dispositivo interruptor? 

e) ¿Qué hace la instrucción RETI? ¿En qué se diferencia de 
un retorno de subrutina? 

f) ¿Cómo podría solucionarse una situación de desbordamien- 
to de la pila? 

g) ¿Qué tiempo de servicio (“tiempo perdido”) introduce el 
mecanismo de interrupción? 


EJECUTAR 
INSTRUCCION 


¿MASCARA? 


SIGUIENTE 
INSTRUCCION 


(si fuese necesario) 
(si fuese necesario) 
Figura 6.35 


Lógica de una interrupción. | RETURN 


ШИП 
НЕНЬ 


чи 


пос d 
Lud 
dis 
ж 
зв 
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Introducción 


Dispositivos 
de entrada/ 
salida 


Ya sabemos programar el microprocesador Z80 en la mayor 
parte de las situaciones habituales, pero es preciso mencionar 
las pastillas de entrada/salida conectadas normalmente al mis- 
mo. Los constantes progresos de integración LSI (integración a 
gran escala) han determinado la introducción de circuitos antes 
inexistentes, de manera que la programación de un sistema 
supone, en primer lugar, la del propio microprocesador y, en 
segundo lugar, la de las pastillas de entrada/salida. De hecho, 
muchas veces es más difícil recordar cómo deben programarse 
las diferentes opciones de control de las pastillas de E/S que 
programar el microprocesador, y no porque la programación en 
sí sea más complicada, sino porque cada uno de esos dispositi- 
vos tiene sus peculiaridades. Examinaremos aquí el tipo más 
general —el dispositivo de entrada/salida programable o PIO—, 
y a continuación veremos algunos dispositivos E/S de Zilog. 


El “PIO estándar” 


No existe el “PIO estándar”, pero todos los fabri- 
cantes de las distintas marcas con el mismo fin funcionan bá- 
sicamente igual. La finalidad de un PIO es proporcionar una 
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Figura 7.1 
Un PIO típico. 


conexión multipuerta a los dispositivos de entrada/salida (una 
puerta no es más que un juego de $ líneas de entrada/salida). 
Cada PIO proporciona al menos dos juegos de líneas de 8 bits 
a los dispositivos de E/S. Estos necesitan siempre una memoria 
auxiliar para estabilizar el contenido del bus de datos, por lo 
menos a la salida. Los PIO dispondrán, por tanto, de al menos 
una memoria auxiliar por puerta. 

Como ya se ha dicho, el microprocesador utiliza para 
comunicarse con los dispositivos de E/S un mecanismo de 
acoplamiento o de interrupción; los PIO se comunican con los 
periféricos de forma parecida, y para ello disponen de un 
mínimo de dos líneas por puerta para efectuar la función de 
acoplamiento. 

El microprocesador necesita saber el estado de cada una de 
las puertas, para lo que, a tal fin, cuenta con uno o más bits de 
estado. Por último, cada PIO dispone de una serie de recursos 
que configuran sus posibilidades. Para especificar las opciones 
de programación, el usuario tiene que acceder a un registro 
especial interno del PIO llamado registro de control; en algunos 
casos, la información de estado forma parte de este registro, 

Esencial a cualquier PIO es la posibilidad de configurar 
cada una de las líneas como de entrada o de salida. La figura 
7.1 recoge el esquema de un PIO. Todas las puertas disponen 
de un registro de dirección de datos que sirve para programar la 
dirección de las líneas. En muchos PIO, un bit “0” en una 


. posición de ese registro significa entrada, y un “1”, salida. Zilog 


utiliza la convención contraria. 


е E с 
Dm noël [383 
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BUSIE 25 Уда пра PUERTA А 

25 992 002 
оо 9659 ono 
= е Ф 


SELECTOR А50 FORAS 
DE REGISTROS 
IROA Ш--св2 
IROB Сві 


Quizá resulte sorprendente utilizar "0" рага la entrada у 
“1” para la salida, cuando, en realidad, es más natural adjudi- 


car “0” a la salida (Output en inglés) y “1” a la entrada (Input); 
pero se trata de una elección deliberada: cuando se aplica al 
sistema la alimentación, es muy importante que todas las líneas 
de E/S estén en configuración de entrada, porque en caso 
contrario, si el microprocesador estuviese conectado a algún 
periférico peligroso, podría activarlo accidentalmente. Cuando 
se aplica la señal de reiniciación (reset), normalmente todos los 
registros pasan a 0, de manera que todas las líneas del PIO 
quedan configuradas como entradas. Las conexiones con el 
microprocesador están representadas a la izquierda de la figura 
mencionada. Como es natural, el PIO se conecta al bus de 
datos de 8 bits, al de direcciones y al de control del microproce- 
sador. El programador no tiene más que especificar la dirección 
de cualquiera de los registros del PIO al que desee acceder. 


REGISTRO INTERNO DE CONTROL 


Este registro cuenta con una serie de opciones de generación 
y detección de interrupciones o de ejecución automática de la 
función de acoplamiento. No es necesario hacer aquí una 
descripción completa de sus recursos; el usuario del sistema no 
tiene más que consultar la hoja de características, que recoge el 
efecto de activar los diferentes bits del registro de control. Cada 
vez que se inicialice el sistema, el usuario deberá cargar el 
mencionado registro del PIO con el contenido adecuado a la 
aplicación en uso. 


Programación de un PIO 


La siguiente sería una secuencia típica de empleo de un 
canal de PIÓ (suponiendo una entrada): 


Carga del registro de control 


Se efectúa mediante una transferencia programada entre un 
registro del Z80 (por lo general, el acumulador) y el registro de 
control del PIO. De esta: forma quedan activadas las opciones y 
el modo de funcionamiento del PIO (véase figura 7.2). Normal- 
mente sólo se hace una vez, al principio del programa, 


Carga del registro de dirección 


Especifica la dirección en que deben utilizarse las líneas de 
E/S (véase figura 7.3). 
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Figura 7.2 
Empleo de un PIO: carga del 
registro de control. 


Figura 7.3 
Empleo de un PIO: carga de la 
dirección del dato. 


DIRECCION 
Lo) DE Los DATOS 
5, 


[TT] | CONEXION 
A] PERIFERICOS А 


CONTROL 


SELECCION CONEXION 
DE PASTILLA ич PERIFERICOS В 


SELECCION 
DE REGISTRO DIRECCION 
DE LOS DATOS 


EE 
| 


CONEXION 
Gu -PA7 
PERIFERICOS A БАРА 
CONTROL 


SELECCION CONEXION (A PB0-PB7 
DE PASTILLA LA] PERIFERICOS B 


SELECCION 
DE REGISTRO DIRECCION 
DE LOS DATOS 


Lectura del estado 


El registro de estado indica si hay o no un byte válido a la 
entrada (véase figura 7.4). 


. REGISTRO DEL | 


| BUS DE DATOS | 
DIRECCION 
Bes emn. 
CONEXION 
PERIFERICOS A 
CONTROL 
SELECCION CONEXION 
DE PASTILLA ғ PERIFERICOS В 
| SELECCION 
1 DE REGISTRO DIRECCION 
DE LOS DATOS 
Figura 7.4 
Empleo de un PIO: lectura del 
estado. 
Lectura de la puerta 
El byte se lee en el Z80 (véase figura 7.5). 
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Figura 7.6 
Patillas del PIO del 280. 


El PIO Zilog Z80 


Es un dispositivo de dos puertas, de arquitectura esencial- 
mente compatible con la que acabamos de describir como 
estándar. La disposición de las patillas aparece en la figura 7.6 y 
el diagrama de bloques en la 7.7. 
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Cada una de las puertas del PIO tiene seis registros: uno de 
entrada, de 8 bits; otro de salida, también de 8 bits; uno de 
control en modo, de 2 bits; uno de máscara, de 8 bits; uno de 
selección entrąda/salida (dirección), de 8 bits, y uno de control 
de la máscara, de 2 bits. Los tres últimos sólo se usan cuando 
la puerta está programada para funcionar en modo bit. 

Las puertas tienen cuatro modos de funcionamiento, que se 
seleccionan mediante el registro de control en modo de 2 bits: 
salida de byte, entrada de byte, bus bidireccional y bit. 

Los dos bits del registro de control de la máscara los carga 
el programador y especifican el estado alto o bajo del periférico 
que debe controlarse y las condiciones en que puede generarse 
una interrupción. 

El registro selector de entrada/salida de 8 bits permite 
disponer cada una de las patillas en dirección de entrada o de 
salida cuando se opera en modo bit. 


PROGRAMACION DEL PIO ZILOG 


Una secuencia para usar el PIO en modo bit, por ejemplo, 
sería: 


Cargar el registro de control en modo para especificar el 
modo bit. 

Cargar el registro selector de entrada/salida de la puerta A, 
para especificar que las líneas 0-5 son entradas y las 6 y 7 
salidas. 

А continuación podría leerse una palabra leyendo el conte- 
nido de la memoria auxiliar de entrada. 
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Figura 7.7 
Diagrama de bloques del PIO 
280. px 


Resumen 


Además, podría utilizarse el registro de filtro para especificar 
las condiciones de estado. 

El lector interesado en una descripción detallada del funcio- 
namiento del PIO deberá consultar el libro de esta misma 
colección Z80 Applications Book. 


El SIO 280 


El SIO (entrada y salida en serie) es un periférico de dos 
canales construido para facilitar las comunicaciones asíncronas 
en serie. Contiene un receptor-transmisor asíncrono universal 
(UART), y su función principal es efectuar la conversión de serie 
a paralelo, y «viceversa. No obstante, la pastilla dispone de 
recursos muy refinados, como la manipulación automática de 
protocolos complejos organizados por bytes, como IBM bisin- 
crono, o HDLC y SDLC, dos protocolos organizados por bits. 

Además, puede funcionar en modo síncrono, como un 
USRT, y generar y verificar códigos CRC. Dispone de los 
modos muestreo, interrupción y transferencia de bloques. La 
descripción completa de este dispositivo está fuera del alcance 
de este texto introductorio, y aparece en el mencionado Z80 
Applications Book. 


OTRAS PASTILLAS DE E/S 


Dado que el Z80 se utiliza habitualmente como sustituto del 
8080, se ha construido de forma que pueda combinarse con casi 
todas las pastillas de E/S de éste, además de con las suyas 
especificas fabricadas por Zilog. Todas las pastillas de E/S 8080 
pueden considerarse compatibles con un sistema 780. 


Para sacar partido a los componentes de entrada/salida hay 
que conocer perfectamente la función de cada bit o grupo de 
bits de los diversos registros de control. Estas nuevas pastillas, 
de estructura compleja, automatizan muchas operaciones que 
antes se confiaban al soporte lógico o a alguna lógica especial. 
Más en concreto, los componentes como el SIO automatizan 
muchas de las operaciones de acoplamiento. Con la informa- 
ción de este capítulo, el lector estará en condiciones de entender 
las funciones de las señales y los registros básicos. Como es 
natural, seguirán apareciendo nuevos componentes que confia- 
rán al soporte físico la ejecución de algoritmos todavía más 
complicados. 
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Aplicaciones 


Introducción 


Este capítulo presenta una serie de programas de aplicación 
práctica pensados para poner a prueba sus conocimientos de 
programación. Estos programas o “rutinas” se encuentran con 
frecuencia en aplicaciones más amplias y se conocen como 
“rutinas de servicio”. Su creación obliga a una sintesis de los 
conocimientos y técnicas adquiridos en los anteriores capítulos. 

Tendremos ocasión de tomar caracteres de dispositivos de 
E/S para someterlos a distintos tratamientos, pero antes de 
llegar a eso aprenderemos a borrar un área de la memoria (cosa 
que no siempre será necesaria; téngase en cuenta que estos 
programas se presentan exclusivamente como ejercicios de pro- 
gramación). 


Borrado de una sección de memoria 


Deseamos borrar —igualar a cero— los contenidos de la 
memoria comprendidos entre las direcciones BASE y BASE 
+ LONGITUD, siendo LONGITUD inferior a 256. El pro- 
grama es: 
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CEROM LD В, LONGITUD CARGAR LONGI- 


TUD EN B 

LD А, 0 BORRAR А 

LD HL, BASE APUNTADOR A 
LA BASE 

BORRAR LD (HL), A BORRAR LA POSI- 

CION A 

INC HL APUNTADOR A 
LA POSICION SI- 
GUIENTE 

DEC В DECREMENTAR 


EL CONTADOR 
JR NZ, BORRAR — ¿FIN DE LA SEC- 
CION? 
RET 


En el programa se supone que la longitud de la sección de 
memoria es igual a LONGITUD. El par de registros HL se 
utiliza como apuntador a la palabra en curso que debe borrar- 
se. El registro B, como es habitual, se emplea como contador. 

El acumulador А se carga una sola vez con el valor 0 (todo 
ceros), y a continuación se copia en posiciones de memoria 
SUCesivas. 

En un programa de verificación de memoria, por ejemplo, 
esta rutina de servicio serviría para poner a 0 el contenido de 
un bloque; a continuación, el programa comprobaría, de la 
forma habitual, si su contenido sigue siendo 0. 

Lo que acabamos de ver es la ejecución normal de una 
rutina de borrado, que todavía puede mejorarse, como demues- 
tra esta nueva versión: 


CEROM LD B, LONGITUD 
LD HL, BASE 
BUCLE LD (HL), 0 
INC HL 
DINZ BUCLE 
RET 


Se han introducido dos mejoras, que consisten en la elimina- 
ción de la instrucción LD A,0, en la carga de un “0” directa- 
mente en la posición señalada por H y L y en el uso de la 
instrucción especial del 780 DJNZ. 

Este ejemplo pone de relieve que la primera versión de un 
programa, aunque sea perfectamente correcta, casi siempre puede 
mejorarse estudiándola atentamente. Para introducir mejoras es 
imprescindible estar muy familiarizado con todas las instruccio- 


nes. Es importante señalar que las mejoras no son sólo cosméti- 
cas, sino que realmente aumentan la velocidad de funcionamien- 
to del programa, reducen el número de instrucciones y, por 
tanto, el espacio de memoria, y, por lo general, aumentan de 
paso la legibilidad; por consiguiente, la facilidad de corrección 
del mismo. | 


Ejercicio 8.1: Escriba un programa de verificación de memoria que 
iguale а 0 un bloque de 256 palabras y que a continuación 
compruebe si hay un O en cada posición. Acto seguido escribi- 
rá todo unos, y volverá a comprobar el contenido del bloque. 
Hecho esto repetirá el proceso escribiendo 01010101 y 
10101010. 


Ejercicio 8.2: Modifiquese el programa del ejercicio anterior en el 
sentido de ocupar la sección de memoria primero todo con 
ceros y a continuación todo con unos. 


Vamos ahora a hacer un muestreo de los dispositivos de 
E/S para averiguar cuáles necesitan servicio. 


Muestreo de dispositivos de E/S 


Supondremos que tales dispositivos están conectados a 
nuestro sistema. Sus registros de estado se encuentran en las 
direcciones ESTADO1, ESTADO2 y ESTADOJ3. El programa 
es el siguiente: 


PRUEBA IN A,(ESTADO1) LEER ESTADO! 
E/S 
BIT 7A TEST BIT "LISTO" 
(BIT 7) 
JP NZ, UNO SALTAR A MANI- 
PULADOR 1 


IN  A(ESTADOZ LO MISMO PARA 
DISPOSITIVO 2 


BIT 7, А 

JP NZ, DOS 

IN A,(ESTADO3) LO MISMO PARA 
EL DISPOSITIVO 
3 

BIT 7, А 

JP NZ, TRES 


(salida sin resultado) 


Como resultado de la instrucción BIT, el bit Z de las bande- 
ras de estado se activa a 1 si el ESTADO es 0. JP NZ (salto si 
no es igual а 0) provoca una bifurcación a la rutina ENE 
correspondiente. 


Introducción de caracteres 


Supongamos que acabamos de observar que hay un ca- 
rácter listo en el teclado; vamos a acumular varios más en un 
área de memoria llamada DEPOSITO hasta que encontremos 
uno especial, SPC, cuyo código ya ha sido definido. 

La subrutina TOMCAR toma un carácter del teclado (véase 
el capítulo 6 para más detalles) y lo deposita en el acumulador. 
Supongamos que no pueden tomarse más de 256 caracteres sin 
que aparezca SPC. 


SERIE LD HL, DEPOSITO SEÑALA AL DE- 
POSITO 
BUCLE САШ TOMCAR TOMAR UN CA- 
RACTER 
CP SPC VERIFICAR $ 
ES EL ESPE- 
CIAL JR 
JR Z, FUERA ¿HALLADO? 
LD (HL), A ALMACENAR 
CAR EN EL DE- 
POSITO 
INC HL SIGUIENTE 
POS DEPOSITO 
JR BUCLE TOMAR SI- 


GUIENTE CAR 
FUERA RET 


Ejercicio 8.3: Tratemos de mejorar esta rutina básica: 
a) Devuélvase el carácter al dispositivo (un teletipo, por 
ejemplo). 
b) Compruébese que la serie de entrada no tiene más de 256 
caracteres. 


Ya tenemos una serie de caracteres en la memoria auxiliar 
(DEPOSITO), que podemos someter a diversos tratamientos. 


Verificación de un carácter 


Vamos a determinar $1 el carácter situado en la posición de 
memoria LOC es igual а 0, a 1 оа 2: 


CUD LD A (LOC) TOMAR UN CARACTER 
00 


CP ¿ES CERO? 

JP  Z,CERO SALTAR A RUTINA 
CP ol ¿ES UNO? 

ІР Z UNO 

CP 02 ¿ES DOS? 

JP Z,DOS 


JP NOVISTO FALLO 


Nos hemos limitado a leer el carácter y a aplicar la instruc- 
ción CP para comprobar su valor. 

Vamos a realizar ahora una berificación de diferente natura- 
leza. 


Verificación de intervalo 


Se trata de determinar si el carácter ASCII que ocupa la 
posición de memoria LOC es una cifra comprendida entre 0 


y 9: 
INTER LD A(LOC) TOMAR UN CARAC- 

TER 

AND ТЕН FILTRAR EL BIT DE 
PARIDAD 

СР 30H ASCUO 

JR C, FUERA ¿CARACTER MUY 
BAJO? 

CP 39H ASCU 9 

JR МС, FUERA ¿CARACTER MUY 
ALTO? 

CP А FORZAR BANDERA 0 


FUERA RET SALIDA 


El carácter ASCII “0” se representa en hexadecimal por “30” 
о por “ВО”, según se use о по bit de paridad. De la misma 
forma, el carácter ASCII “9” se representa como “39” o como 
“Во”. 

El objetivo de la segunda instrucción del programa es bo- 
rrar el bit de paridad 7, si es que se estaba utilizando, para que 
el programa sea aplicable en cualquier caso. А continuación se 
compara el valor del carácter con los ASCII “0” y “9”; al 
emplear una instrucción de comparación, la bandera Z se activa 
a 1 si el resultado es positivo. El bit de acarreo se activa si hay 
acarreo, y se quita en caso contrario. En otras palabras: cuando 
se usa una instrucción CP, el bit de acarreo se activa si el valor 
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del literal que aparece en la instrucción es superior al valor 
contenido en el acumulador, y se pone a “0” si es igual o 
menor. 

La última instrucción, CP A, fuerza un “1” en la bandera Z, 
que se usa para indicar a la rutina de llamada que el carácter 
de CAR está realmente comprendido en el intervalo 0-9. Puede 
emplearse cualquier otra convención, como cargar una cifra en 
el acumulador para indicar el resultado de la comprobación. 


Ejercicio 8.4: ¿Equivale al de arriba el siguiente programa?: 


LD А, (CAR) 


SUB 30H 

JP M, FUERA 
SUB 10D 

JP P, FUERA 
ADD 10D 


Ejercicio 8.5: Determínese si un carácter ASCII contenido en el 
acumulador es o no una letra del alfabeto. 


Observará que en las tablas ASCII se emplea con frecuencia 
la paridad; así, el equivalente del ASCII “0” es “0110000”, un 
código de 7 cifras. Pero trabajando en paridad impar, por 
ejemplo, se garantiza que el número total de unos de una 
palabra es impar; el código pasaría a ser “10110000” (se ha 
incorporado un “1” adicional a la izquierda), que equivale a 
“ВО” en hexadecimal. Vamos, pues, a crear un programa gene- 
rador de paridad. 


Generación de paridad 


Este programa genera paridad par con el bit 7: 


PARIDAD LD A,(CAR) TOMAR UN CA- 
RACTER 
AND ТЕН BORRAR EL BIT 
DE PARIDAD 
JP РЕ, FUERA VERIFICAR SI LA 
PARIDAD YA ES 
PAR 
OR 80H ACTIVAR EL BIT 
DE PARIDAD 
FUERA LD (LOC), A ALMACENAR EL 
RESULTADO 


El programa utiliza el circuito interno de detección de pari- 
dad de que dispone el 780. 

La tercera instrucción —JP PE, OUT— comprueba si la 
paridad de la palabra del acumulador es ya par o no; el 
resultado es positivo si la respuesta es afirmativa (PE), lo que 
da lugar a la salida (FUERA). 

Si la paridad no es par, es decir, si la instrucción de salto no 
se ejecuta, significa que es impar y que debe escribirse un “1” en 
el bit 7, operación de la que se encarga la cuarta instrucción: 


OR 80H 


Por último, el valor resultante se guarda en la posición de 
memoria LOC. 


Ejercicio 8.6: Con el circuito interno de detección de paridad, el 
problema anterior resulta demasiado sencillo de resolver. A 
modo de ejercicio, trate ahora de solucionarlo sin contar con 
dicho circuito; desplace el contenido del acumulador y cuente 
el número de unos para determinar el bit que debe escribirse 
en la posición de paridad. 


Ejercicio 8.7: Con el programa anterior como ejemplo, verifique la 
paridad de una palabra. Debe calcular la paridad correcta y 
compararla con la esperada. 


Conversión de código: ASCII a BCD 


Pasar el código ASCIT al BCD es muy sencillo. Observará 
que la representación hexadecimal de los caracteres ASCII com- 
prendidos entre 0 y 9 va de 30 a 39 o de BO a B9, dependiendo 
de la paridad. La representación BCD se obtiene simplemente 
añadiendo el “3” o la “B”, es decir, filtrando el nibble (cuatro 
bits) de la izquierda: 


ASCBCD CALL INTER COMPROBAR $1 

EL CAR ESTA EN- 
TRE 0 Y 9 

JP NZ,ILEGAL SALIR SI CAR ES 
ILEGAL 

AND ОЕН FILTRAR NIBBLE 
SUPERIOR 

LD (BCDCAR) A ALMACENAR EL 
RESULTADO 
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Conversión de 


Ejercicio 8.8: Escríbase un programa para pasar de BCD a 
ASCII. 


Ejercicio 8.9: Escribase un programa para pasar de BCD a bina- 
rio (es un problema más difícil de resolver). 


Un consejo: el BCD М, М, М, No equivale al binario 
(№ x 19) + Na) x 10 + №) x 10 + No. 

Para multiplicar por 10 se hace un desplazamiento a la 
izquierda (= x2), otro más (= x4), una instrucción ADC 
(= x5) y otro desplazamiento a la izquierda (= x10). 

En notación BCD completa, la primera palabra puede con- 
tener la cuenta de cifras BCD, el nibble siguiente el signo y cada 
uno de los demás nibbles una cifra BCD (suponemos que no 
hay coma decimal). El último nibble del bloque puede permane- 
cer sin utilizar. 


hexadecimal a ASCII 


“A” contiene una cifra hexadecimal, y no tenemos más que 
añadir un “3” (o una “B”) al nibble izquierdo: 


AND OFH CERO EN EL NIBBLE IZ- 
QUIERDO (OPCIONAL) 

ADD  A,30H ASCII 

CP ЗАН ¿NECESARIA CORRECCION? 

JP M, FUERA 

ADD А,7 CORRECCION DE A A F 


Ejercicio 8.10: Realícese la conversión de HEX a ASCII supo- 
niendo que se trabaja en formato empaquetado (dos cifras 
hexadecimales en A). 


Búsqueda del elemento mayor de una tabla 


La dirección de partida de la tabla se encuentra en la 
dirección de memoria ВАЗЕ. La primera entrada de la mis- 
ma es el número de bytes que contiene. El programa se encar- 
gará de buscar el mayor elemento de la misma, del que deposi- 
tará el valor en A y la posición en la dirección de memoria 
INDICE. 

El programa utiliza los registros A, F, B, H y L, y utiliza 
direccionamiento indirecto para poder buscar la tabla en cual- 
quier lugar de la memoria (véase figura 8.1). 


El mayor elemento de una ta- 


A MAX EN CURSO 


B CONTADOR 


MAX LD 
LD 
LD 
INC 
LD 

BUCLE СР 
JR 


LD 

LD 
CAMBIO INC 
DEC 


JR 
RET 


HL, BASE 

B, (HL) 

А, 0 

HL 
(INDICE). HL 
(HL) 

NC, CAMBIO 


A, (HL) 
(INDICE), HL 
HL 

B 

NZ, BUCLE 


APUNTADOR 
A — ——] 


| CUENTA-N | — 
Не 1 


ELEMENTO М 


INDICE 


BASE 


DIRECCIONES 
CRECIENTES 


DIRECCION DE LA 
TABLA 

NUMERO DE BY- 
TES DE LA TABLA 
BORRAR VALOR 
MAXIMO 
INICIALIZAR INDI- 
CE 
ENTRADA 
GUIENTE 
COMPARAR EN- 
TRADA 

SALTA SI ES ME- 
NOR QUE EL MA- 
XIMO 

CARGA NUEVO 
VALOR MAXIMO 
CARGA NUEVO 
VALOR MAXIMO 
SEÑALA LA EN- 
TRADA SIGUIENTE 
DECREMENTA EL 
CONTADOR 
SIGUE SI NO ES 0 


SI- 


El programa comprueba la entrada enésima; si es mayor 
que 0, раза a А, y su posición se recuerda en INDICE; a 
continuación se comprueba la entrada п - 1, y así sucesivamente. 
El programa funciona con enteros positivos. 
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Ejercicio 8.11: Modifiquese el programa para que funcione tam- 
bién con números negativos en complemento a dos. 


Ejercicio 8.12: ¿Funcionará el programa con caracteres ASCII? 


Ejercicio 8.13: Escribase un programa que clasifique n números en 
orden ascendente. 


Ejercicio 8.14: Escríbase un programa que clasifique n nombres de 
3 caracteres cada uno en orden alfabético. 


Suma de N elementos 


502 


El programa que veremos a continuación calcula la suma de 
16 bits de N entradas positivas de una tabla. La dirección de 
partida de la misma se encuentra en ВАЗЕ y su primera entra- 
da contiene el número de elementos N. La suma de 16 bits se 
deposita en las posiciones de memoria SUMBJ y SUMAL. Si 
requiriese- más de 16 bits, sólo se conservarían los 16 inferiores 
(en tal caso se dice que los superiores se truncan). 

El programa modificará los registros A, F, B, H, L, IX, y 
supone que el número máximo de elementos no pasa de 256 
(véase figura 8.2). 


SUMN LD НІ, ВАЅЕ APUNTA LA 
BASE DE LA 
TABLA 
LD  B,(HL) LEE LA LON- 
GITUD EN EL 
CONTADOR 
SUMIG INC HL APUNTA A 
LA PRIMERA 
ENTRADA 
LD IX, SUMBJ APUNTA AL 
RESULTADO, 
INF 
LD (Х+0),0 BORRA RE- 
SULTADO, 
INFERIOR 
LD (ах+),0 Y SUPERIOR 
BUCLESUM LD  A,(HL) TOMA EN- 
TRADA DE 
LA TABLA 
ADD A,(IX +0) CALCULA 
SUMA PAR- 
CIAL 


LD (IX+0),A LA ALMACE- 
NA EN OTRO 
SITIO 

JR NC,NOACARR COMPRUEBA 
SI HAY ACA- 
RREO 

INC (ІХ 1) SUMA ACA- 
RREO 
AL BYTE SUP 

NOACARR INC HL APUNTA А 

LA SIGUIEN- 
TE ENTRADA 

DEC B DECREMEN- 
TA EL CON- 
TADOR DE 
BYTES 

JR М2, BUCLESUM SIGUE SU- 
MANDO 
HASTA EL FI- 
NAL 

RET 


Se trata de un programa muy sencillo que no precisa de más 
explicaciones. 
Ejercicio 8.15: Modifíquese el programa anterior para: 


a) Calcular sumas de 24 bits. 
b) Calcular sumas de 32 bits. 
c) Detectar cualquier desbordamiento. 


LONGITUD = N 
ELEMENTO 1 


ELEMENTO N 


SUMLO 
SUMHI 


Figura 8.2 
Suma de N elementos. 
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Cálculo del total de control 


El total de control es una cifra o un grupo de cifras que se 
calculan a partir de un bloque de caracteres sucesivos. Dicho 
total se va calculando conforme se van almacenando los datos, 
y se coloca al final de los mismos. Para comprobar la integri- 
dad de un dato, se lee, se vuelve a calcular el total de control y 
se compara con el valor almacenado; si hay discrepancia, es que 
se ha producido un error o un fallo. 

Para calcularlo se emplean varios algoritmos. En este caso 
aplicaremos la operación OR exclusivo a todos los bytes de una 
tabla de N elementos y dejaremos el resultado en el acumula- 
dor. Como es habitual, el inicio de la tabla reside en la direc- 
ción BASE, y la primera entrada de la misma es el número de 
elementos М. El programa modifica los registros А, Е, В, H, L; 
N debe ser inferior a 256. 


TCONT LD HL, BASE CARGA LA DIREC- 
CION DE LA TABLA 


EN HL 

LD В, (НГ) HACE N = LONGI- 
TUD 

XOR A BORRA EL TOTAL 
DE CONTROL 

INC HL APUNTA AL PRI- 
MER ELMENTO 

BUCLE XOR (HL) CALCULA EL TO- 

TAL DE CONTROL 

INC HL APUNTA AL SL 
GUIENTE ELEMEN- 
TO 

DEC B DECREMENTA EL 
CONTADOR 

JR NZ, BUCLE REPITE SI NO HA 
TERMINADO 


LD (ТСОМТ)А CONSERVA EL TO- 
TAL DE CONTROL 
RET 


Cómputo de ceros 


El programa cuenta el número de ceros de nuestra tabla 
habitual y lo deposita en la posición TOTAL. Modifica A, B, C, 
H, L, F. 


CEROS LD 


BUCCERO 10 


NOCERO INC 


DEC 


JR 
LD 
LD 


HL, BASE 
B, (HL) 
C,0 

HL 

А, (HL) 

0 


NZ, NOCERO 
С 


НІ, 


NZ, BUCCERO 
А, С 
(TOTAL), A 


APUNTA A LA 
TABLA 

LEE LA LONGI- 
TUD EN EL 
CONTADOR 
TOTAL A 0 
APUNTA A LA 
PRIMERA  EN- 
TRADA 

TOMA UN ELE- 
MENTO 
ACTIVA BAN- 
DERA 0 

¿ES 0? 

SI LO ES, IN- 
CREMENTA EL 
CONTADOR DE 
CEROS 
APUNTA A LA 
SIGUIENTE EN- 
TRADA 
DECREMENTA 
EL CONTADOR 
DE LONGITUD 


ARCHIVARLO 


Ejercicio 8.16: Modifiquese el programa para contar: 
a) El número de asteriscos (carácter “ж”), 
b) El número de letras del alfabeto. 


Transferencia de bloques 


El programa coge una entrada de cada tres del bloque 
fuente situado en la dirección DESDE y las almacena en bloque 


en la dirección HASTA: 


DCADA3 LD 
LD 


LD 
BUCLE EDI 


HL, DESDE 
DE, HASTA 


BC, TAMAÑO 


с) El número de cifras que hay entre “0” y “9”. 


ACTIVA LOS 
APUNTADORES 


TRANSFEREN- 


CIA AUTOMATI- 
CA 
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INC HL 
INC HL SALTA 2 ENTRA- 
DAS 


JP PE, BUCLE 


Transferencia de bloques en BCD 


Se trata de introducir en la memoria varias cifras BCD, es 
decir, de desplazar varios nibbles (véase figura 8.3). El programa 
aparece a continuación: 


TBCD LD B, CUENTA 
LD HL, BLOQUE 
XOR A A=0 
BUCLE RLD 
DEC HL APUNTA AL BYTE 
SIGUIENTE 
DJNZ BUCLE DECREMENTA EL 


BUCLE HASTA 0 


PEZ 
22222 


"A 
n LA 
оа 
ЖҰ 
ж АА 
a 11 


a 
% 

Бес Ж е 
ж М 
E ОО ЖИЕ ЖЖ 


“7 A 


CUENTA 


Figura 8.3 
Transferencia de bloques en 
BCD; memoria. 


El programa utiliza la instrucción RLD, que todavía no ha- 
bíamos usado. RLD produce una rotación a la izquierda de la 
cifra BCD entre À y (HL). (HL) o M denotan los contenidos de 
las posiciones de memoria apuntadas por H y L. 


MINF va a MSUP. 
MSUP va a AINF. 
AINF va a MINF. 


En este caso, “inf” y “sup” hacen referencia a nibbles de 


4 bits. 


Para utilizar la potente instrucción DJNZ, зе ha empleado 
el registro B como contador de cifras. HL se activa para que 


apunte al principio del bloque. 


А se emplea para almacenar la cifra izquierda de desplaza- 
miento en cada rotación entre dos accesos sucesivos al bloque. 
Por convenio, “0” se introduce en el bloque por la parte 


inferior. 


Comparación de dos números de 16 bits con signo 


IX apunta al primer número М1. 
ГУ apunta а N2 (véase figura 8.4). 


El programa activa el bit de arrastre si NI < №2, y el bit Z 


si NI = №. 
COMP LD  B(IX+1) 


LD А, В 
AND 80H 


JR кг, МЕСМІ 
BIT  7,(Y +1) 


RET NZ 
LD АВ 
CP (+1 
RET NZ 
LD  A,(IX) 
CP (IY) 
RET 

NEGMI ХОВ (IY +1) 
RLA 
RET C 
LD A,B 
CP (IY +1) 
RET NZ 
LD A, (IX) 
CP (IY) 
RET 


TOMA EL SIGNO 
DE МІ 


VERIFICA EL SIG- 
NO, BORRA CY 
INVIERTE NI 


INVIERTE N2 


AMBOS SIGNOS 
POSITIVOS 


BIT DE SIGNO A 
CY 

SIGNOS DIFE- 
RENTES 


AMBOS SIGNOS 
NEGATIVOS 
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Figura 8.4 
Comparación de dos números 
con signo. 


MEMORIA 


| DIRECCIONES 
CRECIENTES 


El programa empieza por comprobar los signos de МІ y 
N2. Si МІ es negativo, se da un salto a МЕСМІ; en caso 
contrario, se ejecuta la parte superior del programa. 

Obsérvese que en la quinta línea se utiliza la instrucción 
BIT para verificar directamente el bit del signo de N2 en la 
memoria: 


BIT 7,(Ү +1) 


Podría haberse hecho lo mismo соп МІ, pero сото el valor 
de éste hace falta rápidamente, es más sencillo leerlo en memo- 
ria y conservarlo en B: 


COMP LD  B,(IX +1) 


Es necesario conservar N1 еп В porque AND puede destruir 
el contenido de A: 


LD А, В 
AND 80H 


Obsérvese que se utiliza un retorno condicional en la li- 
nea 6: 


RET NZ 


Es un recurso poderoso del 780 que simplifica la programa- 
ción. 


Obsérvese que la instrucción de comparación actúa directa- 
mente sobre el contenido de la memoria en modo indexado: 


СР (1Y+1) 


Al comparar los dos números, se empieza por el byte más 
significativo y se pasa a continuación al menos. 

Obsérvese el generoso empleo del mecanismo de indexación 
que se hace en el programa y que da lugar a una codificación 
muy eficaz. 


Ordenación por burbuja 


Es una técnica de ordenación que sirve para organizar los 
elementos de una tabla en orden ascendente o descendente. 
Debe su nombre a que los elementos menores “flotan” por 
entre los demás hasta la parte superior de la tabla. Cada vez 
que un elemento menor “choca” con otro “más pesado”, salta 
por encima del mismo. 

La figura 8.5 recoge un ejemplo práctico de burbuja. La lista 
que debe ordenarse contiene los elementos (10, 5, 0, 2, 100) y 
debe organizarse en orden descendente (de forma que el “0” 
quede en la parte superior). El algoritmo es sencillo y el diagra- 
ma de flujo aparece en la figura 8.7. 

Se comparan los dos elementos superiores (o los dos inferio- 
res); si el inferior es menor (*más ligero") que el superior, se 
intercambian; en caso contrario, se dejan como están. Por razo- 
nes prácticas, el intercambio, si se produce, se recuerda en una 
bandera llamada “CAMBIADO”. La operación se repite con 
los dos elementos siguientes, a continuación con los otros dos y 
así, sucesivamente, hasta haber comparado todos dos a dos. 

El primer paso se ilustra en las fases 1, 2, 3, 4, 5 y 6 de la 
figura 8.5; la operación avanza de abajo arriba, pero igualmente 
podría haberse hecho al revés. 

Si no se produce ningün intercambio, es que la ordenación 
está terminada. Si se produce alguno, hay que empezar de 
nuevo con las comparaciones. 

En la figura 8.6 se observa que el ejemplo propuesto requie- 
re cuatro pasadas. Se trata de un proceso sencillo y muy usado. 

Hay una complicación adicional debida al propio mecanis- 
mo de intercambio. En efecto, al intercambiar À y B no puede 
escribirse 


А-В 
В-А 


Figura 8.5 


Fases 1 a 12 de la ordenación 


por burbuja. 


510 


100 >2: 
SIN CAMBIO 


НЫЕ e 


CAMBIADO 


n 


"if 9 
11 


| 


ВН e 


100 >2: 
SIN CAMBIO 


м 


2 
4 
5% O 
A 
= од 
p». O 5: 
= 
m 
o 


100 >5: 
SIN CAMBIO 


© 


ю 
У 
© 


"i 
[^] 


SIN CAMBIO 


| 


> 

E 

m 

D 

o o 
(22 

E4 

Ei 

o 


^ir e 


0 «10: 
INTERCAMBIO 


2 <5: 
INTERCAMBIO 


CAMBIADO 


НН: е 


5 <10: 
INTERCAMBIO 


о кю 


! 


0 <5: 


CAMBIADO O 


O 


FIN DEL PASO 1 


CAMBIADO 


2>0: 
SIN CAMBIO 


© 


FIN DEL PASO 2 


CAMBIADO 


(3 


Figura 8.6 
Fases 16 a 21 de la ordena- 
ción por burbuja. 


T 


2 2>0: 100 >10: 
SIN AMBIO SIN CAMBIO SIN CAMBIO 


O 


FIN DEL PASO 3 


И 
И 


10>5: 5>2: 2>0: 
SIN CAMBIO SIN CAMBIO SIN CAMBIO 


© 


FIN 


porque ello daría lugar a la pérdida del valor anterior de A 
(pruebe a hacerlo con un ejemplo). 


La solución correcta es utilizar una variable o una posición 
provisionales para conservar el valor de A: 


PROV =A 
A=B 
B = PROV 


Pruebe con un ejemplo, y verá que funciona; esta técnica se 
llama permutación circular. 

Todos los programas realizan el intercambio de esta forma, 
que ilustra el diagrama de flujo de la figura 8.7. 

La distribución de registros aparece en la figura 8.8; el 
programa es el siguiente: 


BURBUJA LD (PROV), HL PROV = (HL) 
OTRAVEZ LD IX, (PROV) IX = (HL) 
RES САМВІО, Н INTERCAMBIO 
BANDERA =0 
LD B.C 
DEC B 
511 


Figura 8.7 
Diagrama de flujo de la ordena- 
ción por burbuja. 
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CAMBIADO = 0 
TOMAR NUMERO 
DE ELEMENTOS N 

ї=М 


LEER ELEMENTO 
Е(І) 
DECREMENTAR | 


¿CAMBIADO = 1? 


NO NO 
INTERCAMBIAR E Y Е 
ROV = Е(!) 
ЕП) = El) 
ЕДІ) = TEMP 
BUCLE LD A, (IX) 
LD D, A D = ENTRADA 
EN CURSO 
LD E, (IX + 1) Е = ENTRADA 
SIGUIENTE 
СР Е COMPARA- 
CION 
JR NC, CAMBIO IR A CAMBIO SI 
EN CURSO > 
SIGUIENTE 


INTER LD (IX), E ALMACENAR 
SIGUIENTE EN 
EL CURSO 
LD (IX + 1),D ALMACENAR 
EN CURSO EN 
SIGUIENTE 
SET BAND, H INTERCAMBIO 
BANDERA = 1 
CAMBIO INC IX ENTRADA SI- 
GUIENTE 
DJNZ SIGUIENTE DECREMENTA 
B, SIGUE HAS- 
TA CERO 
BIT CAMBIO, H ¿INTERCAM- 
BIO = 1? 
JR NZ, OTRAVEZ REINICIA SI 
BANDERA = 1 


INTERCAMBIO/NO 


EN H 
o O ge | 
| SIGUIENTE | | ENCURSO | 


PROV 


Figura 8.8 
Ordenación por burbuja. 


Resumen 


Hemos visto en este capítulo una serie de rutinas de servicio 
que combinan técnicas ya estudiadas en otros anteriores y que 
deben permitir al lector empezar a crear sus propios programas. 
Muchas de las rutinas utilizan una estructura de datos llamada 
tabla, que, como veremos en el capítulo siguiente, no es la única 
que existe. 
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Ld 
E od 
ще 
ж 
ж 
w 
w 
ж 
а 
ed 
Eod 
> 
ж. 


ame 

Ld 
que 
m 
B 
e 
od 
ше 

3: 
ж 
Lud 
3 
ж 


E 
PS 
ae 
е 
ще 
Ld 


ж. 
Ld 
Ld 
ж 
ж 
зв 
ЖЕ 


шы 
жж 
зи: 
bd 
зв: 
зе 
de 


«bla 


Estructuras 
de datos 


PARTE 1: TEORIA 


Introducción 


El diseño de un buen programa supone dos tareas: diseño 
del algorítmo y diseño de las estructuras de datos. En los más 
sencillos no existen estructuras de datos de importancia, por lo 
que la tarea fundamental se reduce a diseñar el algoritmo y 
codificarlo acertadamente en un lenguaje máquina determinado, 
que es lo que hemos hecho hasta el momento. Sin embargo, 
para hacer programas de cierta complejidad es necesario cono- 
cer las estructuras de datos. Dos de ellas ya las hemos usado 
frecuentemente: la tabla y la pila. La finalidad de este capítulo 
es presentar otras más generales que puedan resultar de utili- 
dad. El texto del mismo es completamente independiente del 
microprocesador y del ordenador que se utilice, puesto que su 
contenido es teórico y se refiere a la organización lógica de los 
datos en el sistema. Hay libros dedicados exclusivamente al 
estudio de las estructuras de datos, de la misma forma que los 
hay especializados en eficacia de computación o algorítmos de 
división y otras operaciones habituales; por tanto, nos limitare- 
mos aquí a lo fundamental, sin pretender ser exhaustivos. 
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Punteros 


Un puntero es un número que designa la posición de un 
dato. Todo puntero es una dirección, pero no necesariamente 
toda dirección es un puntero. Para serlo, debe señalar un 
dato o una información estructurada. Ya hemos trabajado fre- 
cuentemente con uno: el puntero de la pila, que señala la 
parte superior de la misma (o la posición situada inmediata- | 
mente por encima de dicha parte superior). Como veremos еп 
seguida, la pila es una estructura de datos muy corriente llama- 
da LIFO (last in, first out: último en entrar, primero en salir). 

Cuando se trabaja con direccionamiento indirecto, la direc- 
ción indirecta es siempre un puntero que señala el dato que 
desea recuperarse. 


Ejercicio 9.1: Estudie la figura 9.1; en la dirección 15 de memoria 
hay un puntero que señala la tabla T, que empieza en la 
dirección 500. ¿Cuál es el verdadero contenido del puntero que 


señala hacia T? 
PUNTERO DE T 


TABLA T 


Casi todas las estructuras de datos están organizadas en 
forma de listas de diferente naturaleza. 


500 


Figura 9.1 
Puntero de dirección. 


Listas 


LISTAS SECUENCIALES 


La lista secuencial también llamada tabla o bloque, es, 
probablemente, la estructura de datos más sencilla (ya utilizada 
en capítulos anteriores). Una tabla suele ordenarse en función 
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Figura 9.2 
Estructura de un directorio. 


de un criterio específico, como el orden alfabético o numérico, 
que facilita la labor de recuperar un elemento de la misma 
(mediante direccionamiento indexado, por ejemplo). Por bloque 
suele entenderse un grupo de datos de límites definidos, pero de 
contenido no ordenado; puede contener series de caracteres о 
ser un sector de un disco o un área lógica de memoria (llamada 
también segmento). En tales casos, no suele ser fácil acceder a 
un elemento aleatorio del bloque. Para facilitar la recuperación 
de bloques de información se utilizan directorios. 


DIRECTORIOS 


Un directorio es una lista de tablas o bloques; los archivos, 
por ejemplo, suelen seguir una estructura de directorio. А modo 
de ilustración, el directorio maestro de un sistema podría conte- 
ner una lista de nombres de usuarios, como la ilustrada en la 
figura 9.2; la entrada del usuario “Juan” señala el directorio del 
archivo Juan, que consiste en una tabla que contiene los nom- 
bres de todos los archivos de Juan y sus posiciones; se trata, 
pues, de una tabla de punteros. En este caso, el directorio es 
de dos niveles, pero un sistema flexible debe permitir la inclu- 
sión de todos los directorios intermedios que puedan resultar 
cómodos para el usuario. 


DIRECTORIO DEL USUARIO 


DIRECTORIO DEL 
FICHERO DE JUAN 


FICHERO DE 
JUAN ALFA 


SIGMA 


LISTA ENCADENADA 


En un sistema suele haber bloques de información que re- 
presentan datos, acontecimientos u otras estructuras que no 
resultan fáciles de desplazar, pero que casi siempre pueden 
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reunirse en una tabla para clasificarlos o estructurarlos. El 
problema que se plantea es que nos interesa dejar todos los 
datos donde están y, a la vez, introducir un orden entre ellos ( 
(primero, segundo, tercero, etc). La solución a este problema 
podría ser una lista encadenada, como la que se ilustra en la | 
figura 9.3. Un puntero de la lista, llamado PRIMERBLO- 
QUE, señala el primer bloque de la misma; dentro de este 
bloque 1 hay una posición reservada —la primera o la última ( 
palabra, por ejemplo— que contiene un puntero orientado 
hacia el bloque 2 y llamado PTR1; el mismo proceso se repite 
para los bloques 2 y 3. Dado que en el ejemplo que nos ocupa 
éste es el último de la lista, PTR3 contendrá, por convenio, o А 
un valor especial “nulo” о un puntero orientado hacia sí | 
mismo que permita detectar el final de la lista. Se trata de una 
estructura económica que sólo exige un puntero por bloque y 
ahorra al usuario la necesidad de tener que desplazar los blo- 


ques en la memoria. 
PRIMER e 
o но: PASSE Е: E. 
Е а 
Lista encadenada. y 


Estudiemos, por ejemplo, la inserción de un nuevo bloque, 
que muestra la figura 9.4. Supongamos que el tal bloque nuevo 
se encuentra en la dirección NUEBLOC y debe insertarse entre 
el bloque 1 y el bloque 2; basta adjudicar al puntero PTRI el 
valor NUEBLOC para que señale hacia el bloque X; PTRX 
albergará el anterior valor de РТКІ y, por tanto, seguirá seña- 
lando al bloque 2. Los demás punteros de la estructura 
permanecen invariables, de modo que para insertar un bloque | 
nuevo basta con actualizar dos punteros de la estructura. 


BLOQUE X 
a. 
= 
BLOQUE 2 
a 


PTR 1 
ет? | 


BLOQUE NUEVO um 


Figura 9.4 PRIMER 


Inserción de un bloque nuevo. BLOQUE 


od e 
BLOQUE 1 BLOQUE 3 Ej | 
а Du 


Ejercicio 9.2: Dibújese un diagrama que represente la extracción 
del bloque 2 de la estructura descrita. 


Se han desarrollado varios tipos de listas que facilitan for- 


mas de acceso, inserción y eliminación específicas, de las que a 
continuación examinaremos las más comunes. 
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Figura 9.5 
Una cola. 


COLA 


La cola se denomina formalmente FIFO (first in, first out: 
primero en entrar, primero en salir), y su estructura se ilustra en 
la figura 9.5. Supongamos, para clarificar el diagrama, que el 
bloque de la izquierda es una rutina de servicio de un dispositi- 
vo de salida (una impresora, por ejemplo). Los bloques de la 
derecha permiten acceder a diferentes programas o rutinas de 
impresión; el orden en que se les atiende viene determinado por 
la cola de espera. Es fácil comprobar que quien primero obten- 
drá servicio será el bloque 1; después, el 2, y a continuación, el 
3. El convenio establece que en una cola el último aconteci- 
miento en llegar se sitúa al final de la misma (detrás de PTR3 
en este caso). De esta forma se garantiza que el primero que se 
insertó en la misma será el primero en ser atendido. En siste- 
mas informáticos es normal organizar en colas los aconteci- 
mientos que pueden esperar la atención de recursos escasos, 
como el procesador o algunos dispositivos de entrada/salida. 


RUTINA DE SERVICIO 


SIGUIENTE 


BLOQUE 1 


PTR1 


BLOQUE 3 


| 


BLOQUE 2 


PILA 


Esta estructura ya la hemos examinado detalladamente a lo 
largo de todo el libro; se denomina LIFO (último en entrar, 
primero en salir), porque el último elemento que se deposita en 
la parte superior de la misma es el primero que se extrae. La 
pila puede materializarse como bloque clasificado o como lista. 
Dado que en los microprocesadores la lista se emplea sobre 
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Figura 9.6 
Lista circular. 
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todo en acontecimientos de alta velocidad, como subrutinas o 
interrupciones, lo que suele alojarse en la misma no es una lista 
encadenada, sino un bloque continuo. 


LISTA ENCADENADA FRENTE A BLOQUE 


Una cola puede también materializarse en forma de bloque 
de posiciones reservadas. El bloque continuo tiene la ventaja de 
que se eliminan punteros y se acelera la recuperación y el 
inconveniente de que hay que reservar un bloque bastante gran- 
de para acomodar la estructura de tamaño más desfavorable 
que se prevea. También es difícil, o poco práctico, insertar y 
extraer elementos del interior del bloque. Como la memoria es 
casi siempre un recurso escaso, los bloques suelen reservarse 
para estructuras de tamaño fijo o que exijan la máxima veloci- 
dad de recuperación, como la pila. 


LISTA CIRCULAR 


Se llama lista circular a una lista encadenada en la que la 
última entrada señala hacia la primera (véase figura 9.6). Nor- 
malmente se lleva también un puntero del bloque en curso. 
Cuando se trata de sucesos о programas, el' puntero de 
suceso en curso se mueve una posición a la derecha o a la 
Izquierda en cada ocasión. La lista circular suele utilizarse cuan- 
do todos los bloques se supone que tienen idéntica prioridad, 
aunque también puede emplearse como caso particular de otras 
estructuras para facilitar la recuperación del primer bloque des- 
pués del último cuando se lleva a cabo una búsqueda. 


SUCESO 1 SUCESO 2 ... SUCESO N 


SUCESO EN CURSO 


Los programas de muestreo funcionan, por lo general, de 
forma parecida a una lista circular: interrogan a todos los 
periféricos, y cuando llegan al último vuelven a empezar por el 
primero. 


ARBOLES 


Siempre que hay una relación entre todos los elementos de 
una estructura (es lo que se llama sintaxis) puede utilizarse la 
estructura de árbol, que es de tipo descendente o genealógico. 


Figura 9.7 
Arbol genealógico. 


Figura 9.8 
Lista doblemente encadenada. 


La figura 9.7 ilustra una situación de este tipo: Jaime tiene 
un hijo llamado Roberto y una hija llamada Juana; ésta, a su 
vez, tiene tres niños: Elisa, Tomás y Felipe; Tomás, por su 
parte, tiene otros dos: Manuel y Cristina; Roberto (a la izquier- 
da de la figura), por el contrario, no tiene descendencia. 


ROBERTO 


ELISA FELIPE 
MANUEL CRISTINA 


La estructura que une esas relaciones es un árbol. La figura 
9.2, que ya hemos examinado al principio de este mismo capítu- 
lo, es también un ejemplo de árbol sencillo. La estructura de 
directorio es un árbol de dos dimensiones. Los árboles se em- 
plean siempre que los elementos pueden clasificarse de acuerdo 
con una estructura fija, que facilita la inserción y la recupera- 
ción de los mismos. Además, el árbol permite establecer grupos 
de información estructurados que pueden ser necesarios en ulte- 
riores tratamientos (por ejemplo, en un compilador o en un 
intérprete). 


LISTA DOBLEMENTE ENCADENADA 


Entre los elementos de una lista pueden establecerse enlaces 
adicionales, de los que los más sencillos son los que aparecen en 
la llamada lista doblemente encadenada, que muestra la figura 
9.8. Como se ve, contiene la estructura habitual de enlaces de 
izquierda a derecha más otra de derecha a izquierda. De lo que 
se trata es de facilitar la recuperación de los elementos situados 
justamente antes y después del que se está procesando, aunque 
para ello es necesario prever un puntero más por bloque. 


с - m a 
BLOQUE 1 BLOQUE 2 iz BLOQUE 3 
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Búsqueda y ordenación 


La búsqueda y la ordenación de los elementos de una lista es 
una Operación que depende directamente de la estructura utili- 
zada para dicha lista. Se han desarrollado numerosos algorít- 
mos de búsqueda para las estructuras de datos habituales, de 
los que ya hemos empleado el direccionamiento indexado. Es 
un método de acceso posible cuando los elementos de la tabla 
están ordenados en función de un criterio conocido; tales ele- 
mentos pueden recuperarse por sus números. 

Se llama búsqueda secuencial a la exploración lineal de un 
bloque completo. Se trata de un método claramente ineficaz, 
pero que, en ocasiones, hay que emplearlo a falta de algo mejor 
si los elementos están totalmente desordenados. 

La llamada búsqueda binaria о logarítmica busca un elemen- 
to en una lista ordenada dividiéndola en dos tras cada explora- 
ción. Si, por ejemplo, se trata de una lista alfabética, la búsque- 
da podría comenzar por determinar si el nombre buscado está 
antes o después de la mitad de la misma; si está después, se 
elimina la primera mitad de la lista y se vuelve a dividir en dos 
la segunda, para repetir en ella la misma operación, y así 
sucesivamente hasta llegar al elemento buscado; de esta forma, 
la longitud máxima que hay que explorar es log,n, siendo n el 
número de elementos de la tabla. Aparte de ésta, hay muchas 
otras técnicas de búsqueda. 


Resumen de la sección 
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En esta sección hemos tratado únicamente de hacer una 
breve presentación de las estructuras de datos que más frecuen- 
temente utiliza el programador. Aunque tales estructuras están 
organizadas por tipos y tienen un nombre, la organización 
general de los datos en sistemas complejos recurre frecuente- 
mente a combinaciones de varias de ellas u obliga al programa- 
dor a inventar otras nuevas adecuadas al fin perseguido; así, las 
posibilidades sólo están limitadas por la imaginación del pro- 
gramador. De la misma manera, se han desarrollado una serie 
de técnicas de ordenación y búsqueda para las estructuras de 
datos más habituales, aunque su descripción está fuera del 
alcance de este libro. La finalidad de esta sección es subrayar la 
importancia que tiene el diseño de estructuras adecuadas para 
los datos que van a manipularse y proporcionar las herramien- 
tas necesarias para satisfacer ese objetivo. A continuación vere- 
mos ejemplos más detallados de su aplicación a programas 
reales. 


i 


PARTE Il: EJEMPLOS PRACTICOS 


Introducción 


Veremos aquí algunos ejemplos reales de diseño de estructu- 
ras de datos típicas: tabla, lista clasificada y lista encadenada, 
junto con los algorítmos de búsqueda, inserción y borrado 
correspondientes a las mismas. 

A] lector interesado en técnicas avanzadas de programación 
le resultará de utilidad el estudio minucioso de los programas 
reunidos en esta sección. 

Por el contrario, el principiante puede prescindir de su estu- 
dio en un principio y volver sobre ello cuando se considere más 
preparado. | 

Para seguir los ejemplos de esta parte es imprescindible 
entender perfectamente los conceptos presentados en la anterior. 
Además, los programas utilizarán todos los modos de direccio- . 
namiento del Z80 e integrarán muchas de las ideas y técnicas 
estudiadas en anteriores capítulos. 

Estudiaremos aquí una lista sencilla, una alfabética y una 
lista encadenada con directorio. Para cada estructura vamos a 
desarrollar tres programas: buscar, introducir y eliminar. 
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Representación de datos en la lista 


Tanto en la lista sencilla como en la alfabética representare- 
mos los elementos de la misma forma: 


ЕЛЕЛЕЛЕЛЕЛ ӨЗЕЛЕЛ 


———ы— 
etiqueta de 3 bytes Datos 


ENTLEN LONGITUD DE LA ENTRADA 


TABLEN м] NUMERO DE ENTRADAS 


BASE DE LA TABLA 
ETIQUETA 
ENTRADA 
M BYTES 


Figura 9.9 
Estructura de la tabla. 


«i-— —— INTRODUCCION DE UN 
NUEVO ELEMENTO 


Cada elemento o “entrada” consta de una etiqueta de 3 
bytes y un bloque de datos de n bytes, estando n comprendido 
entre 1 y 253, de manera que cada uno utiliza, como máximo, 
una página (256 bytes). Todos los elementos de la lista tienen la 
misma longitud (véase figura 9.10). Los programas que manipu- 
lan estas dos sencillas listas comparten una serie de convencio- 
nes: 


ENTLEN es la longitud de un elemento. Así, si cada uno 
tiene 10 bytes de datos, ENTLEN vale: 3 + 10 = 13. 

TABASE es la base de la lista o tabla de memoria. 

POINTR es el apuntador móvil del elemento en curso. 

OBJETO es la entrada en curso de localización, inser- 
ción o eliminación. 

TABLA es el número de entradas. 


"m 


524 


Figura 9.10 : 
Entradas típicas de la lista en 
memoria. 


Lista sencilla 


ETIQUET, 


1 


ENTLEN 


| 
| 
| 


ELEMENTO 
2 


ENTLEN 


DATOS 


Se supone que todas las etiquetas son diferentes. De todas 
formas, bastan modificaciones mínimas de los programas para 
cambiar estas convenciones. 


La lista se organiza como tabla de n elementos no claisifca- 
dos (véase figura 9.11). Para buscar uno hay que recorrer la 
lista hasta dar con él o hasta llegar al final de aquélla. La 
inserción se realiza añadiendo nuevos elementos a los ya exis- 
tentes. Cuando se elimina alguno, los situados en las posiciones 
superiores de la memoria —en caso de que haya alguno— se 
desplazan para cubrir los huecos y que la tabla sea continua. 


BUSQUEDA 


Se utiliza una técnica de búsqueda en serie que consiste en 
comparar las etiquetas una por una y letra por letra con la de 
OBJETO. А 

El puntero móvil POINTR se inicializa al valor de 
TABASE. El diagrama de flujo del algoritmo de búsqueda, que 
avanza de forma obvia, aparece en la figura 9.12, El programa 
se encuentra en la figura 9.16, situada al final de esta sección 
(programa “ВОЅСА”). La figura 9.17 contiene un pase de prue- 
ba del mismo. 
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TABASE 


ELEMENTO 1 
ELEMENTO 2 
ELEMENTO 
EN CURSO 

ELEMENTO n 
ESPACIO LIBRE 


OBJETO QUE 
VA A INSERTARSE 


LONGITUD = 
ENTLEN 


POINTR 


(TABLEN = n) 


ESPACIO LIBRE INSERCION 


Figura 9.11 
Lista sencilla. 


BUSQUEDA 


ONTADOR = NUMERO) 
DE ENTRADAS 


¿CONTADOR = 0? 


NO 
LEER ENTRADA 
(3 LETRAS) 


LISTA VACIA 


HALLADA 
(PONER A А “FF”) 


NO 
CONTADOR = 
CONTADOR — 1 


5! SALIDA EN CASO 


¿CONTADOR = 0? DE ERROR 


OTERO АГА 


. P 
Figura 9.12 ENTRADA SIGUIENTE 


Diagrama de flujo de la bús- 
queda en la tabla. 
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Figura 9.13 
Diagrama de flujo de inserción 
en la tabla. 


INSERCION 


Para insertar un nuevo elemento se utiliza el primer bloque 
de bytes disponible en memoria (ENTLEN) al final de la lista 
(véase figura 9.11). 

El programa empieza por comprobar si la nueva inserción 
no está ya en la lista (en este ejemplo se supone que todas las 
etiquetas son diferentes). Si no está, incrementa la longitud de la 
lista TABLEN y lleva OBJETO al final de la misma. El corres- 
pondiente diagrama de flujo aparece en la figura 9.13. 


¿ESTA EL OBJETO DENTRO? SALIDA 


GUARDAR ANTERIOR LONGITUD 
DE LA T, 


INCREMENTAR LONGITUD 
DE LA TABLA 


5 
> 


ем 


El programa completo aparece en la figura 9.16; se llama 
“NUEVO” y reside en las posiciones de memoria 013D a 0166. 

El registro de índice ГУ señala la fuente. HL y DE son los 
apuntadores de destino. { 


ELIMINACION 


Para eliminar un elemento de la lista no hay más que subir 
una posición los situados a continuación en direcciones superio- 
res y decrementar la longitud de la lista. La operación se ilustra 
en la figura 9,14, 

El programa es bastante sencillo, y aparece en la figura 9.16. 
Se llama “BORRAR” y reside en las direcciones de memoria 
0167 a 018Е; el diagrama de flujo está en la figura 9.15. 
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ANTES DESPUES 


TEMPTR 
Figura 9.14 MOVER | 
Eliminación de un elemento | 
(tabla sencilla). | 


La posición de memoria TEMPTR ез un puntero provi- 
sional que señala el elemento que ha de moverse hacia arriba. 
Durante la transferencia, POINTR señala siempre el hueco 

de la lista, es decir, el destino de la transferencia de bloque | 
siguiente. | 
| 


ELIMINAR 
MOVER 


La bandera Z se usa a la salida para indicar que el resulta- 
do de la eliminación ha sido positivo. 

Obsérvese que la instrucción LDIR ejecuta una transferencia 
de bloque automática eficaz (véase la dirección 0178 en la figura 


9.16). 
LD А, В CONTADOR DE 
BLOQUE 
BLONUE LD BC,(ENTLEN) LONGITUD DEL 
BLOQUE | 
LDIR | 
DEC А | 
JP NZ, BLONUE 
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Figura 9.15 
Diagrama de flujo de la elimi- 
nación de la tabla. 


BUSCAR ENTRADA 


¿HALLADA? SALIDA 


si 


DECREMENTAR LA LONGITUD 
DE LA TABLA 


HALLAR NUMERO DE ENTRADAS 
TRAS EL OBJETO 


SALIDA 


NO 


DESPLAZAR UNA ENTRADA 
HACIA ARRIBA 


DECREMENTAR LA CUENTA 
DE ENTRADAS QUE QUEDAN 
TRAS LA DESPLAZADA 


SALIDA 
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Figura 9.16 
Programas de la lista sencilla. 
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9100 
0100 
0102 
0104 
0106 


0108 
010A 
0100 
010Е 
010Е 
0110 
0114 
0117 
011A 
0110 
0120 
0125 
0126 
0129 
O12C 


012F 


0150 
0151 


0155 
0157 
O13A 


015С 


0150 
0140 
0141 
0144 
0147 
0148 
0149 
014С 
O14E 
0151 


0155 
0156 
0157 
0159 
0150 
O1SF 
0160 
0161 


0163 
0166 


0167 
о16а 
016В 
016Е 
0171 
0172 
0175 


0176 
0179 
017В 
017С 


017F 
0180 
0181 


8F01 
9101 
9201 
9401 


1500 
3A0201 


DD240401 
DD7EOO 
FDBEOO 
C22F01 
DD7EO1 
FDBEO1 
C22F01 
DD7E02 
FDBEO2 
САЗАО1 


os 


ca 
EDSBOO01 


DD19 
C31401 
16FF 


c? 


CD0801 


Е0480001 


41 

19 

1020 
Е0480001 


O1FFFF 
C9 


CDO801 
14 
C28B01 
3A0201 
3D 
320201 
95 


CABBO1 
DDES 
01 
280001 


19 
78 
ED4B0001 


ENTLEN 
TABLEN 
TABASE 
TEMP 


, 
BUSCA 


BUCLE 


LI 
SIGUIE 


ENCONT 


UEVO 


BUCLEE 


FUERAE 


Wu v 


ORRAR 


BLONUE LD 


#0100 
ENDER 
ENDER+2 
ENDER+S 
ENDER+S 


D,0 
A, (TABLEN) 
A 

7 

B,A 

IX, (TABASE) 
A, (1Х+0) 
(IY+0) 

NZ, SIGUIE 
А, (1Х+1) 
(1Y+1) 

№7, SIGUIE 
A, (1Х+2) 
(1у+2) 

Z, ENCONT 


B 


1 
DE, (ENTLEN) 


IX,DE 
BUCLE 
D, НОЕЕ 


BUSCA 
D 

2,FUERAE 

A, (TABLEN) 
Е, А 

А 
(TABLEN),A 
о, о 

HL, (TABASE) 
BC, (ENTLEN) 


B,C 

HL,DE 
BUCLEE 

BC, (ENTLEN) 
1 

DE 

DE, HL 


E 


BC, HOFFFF 


BUSCA 

D 

NZ, SALIDA 
A, (TABLEN) 
A 
(TABLEN),A 
B 


7, SALIDA 

IX 

DE 

HL, (ENTLEN) 


HL,DE 
A,B 
BC, (ENTLEN) 


3 INICIALIZA D 
3 COMPRUEBA TABLA LONGITUD O 
3DEFINE FLAGS 


3 ALMACENA LONGITUD TABLA 
3PONE DIRECCION BASE EN IX 
3 COMPRUEBA PRIMERA LETRA 


3 COMPRUEBA SEGUNDA LETRA 


3COMPRUEBA TERCERA LETRA 


SALIDA SI TODAS LETRAS 
; VALIDAS 

3 DECREMENTA CONTADOR 
;LONGITUD TABLA 

¿SALIDA SI FINAL DE TABLA 
;DEFINE EN IX LA SIGUIENTE 
¿DIRECCION DE ENTRADA 


3PRUEBA OTRA VEZ 
3FIJA EN D LA DIRECCION 


3DE ENTRADA EN LA TABLA 
¿CONTENIDA EN IX 


¿VE SI EL OBJETO ESTA ALLI 

3SI D ERA FF SALIR 

3CARGA E CON LONGITUD TABLA 

3 INCREMENTA LONGITUD TABLA 

¿FIJA В A LA LONGITUD DE 

3UNA ENTRADA 

3 зима HL A (ENTLEN*TABLE) 

¿MUEVE IY A DE \ 
3MUEVE LA MEMORIA DEL 


; OBJETO AL FINAL 
БЕ LA TABLA 


¿LOCALIZA ENTRADA A BORRAR 
¿VE SI SE HA ENCONTRADO 


3 DECREMENTA LONGITUD TABLA 


3 AHORA B=NUMERO DE ENTRADAS 
3 DEJADAS EN LA TABLA 
3...DESPUES DE BORRAR UNA 
¿MUEVE IX A DE 


¿COLOCA HL UNA ENTRADA 
¿DELANTE DE DE 


¿FIJA EL CONTADOR DE BLOQUE 
¿FIJA EL CONTADOR DE 


Figura 9.16 


Programas de la lista sencilla 


(continuación). 


Figura 9.17 


Pase de prueba de los progra- 
mas de la lista sencilla. 


0185 Е 


0187 $ 
0188 C 


DBO 


D 
28101 


780 
790 
BOO ; 
810 
820 


e. 


830 SALIDA 
840 FUERA 


850 


, 
860 ENDER 


BORRAR 
BUCLEE 
ENCONT 
ENTLEN 
FUERAE 


0167 
0156 
013A 
0100 
0166 


LDIR ; 


DEC A 

JP М7, BLONUE 
LD BC, #OFFFF 
RET 


END 


¿LONGITUD DE BLOQUE 
¿SHIFT DE UNA ENTRADA DE 
JLA TABLA 


¿SHIFT DE OTRO BLOQUE 
¿DEMUESTRA QUE ESTA HECHO 


NUEVO 0150 
SALIDA O18B 
TABASE 0104 
0106 


TEMP 


Distribución de la memoria 


-имз00 


0300 
0310 
0320 
0330 
0340 
0350 
0360 
0370 


-SY 
Y=0000 300 


-6193/196 


53 
44 
4D 
55 
41 
00 
00 
00 


4F 4Е 31 
41 44 32 
4F AD 33 
4E 43 34 
4Е 54 35 
00 00 00 
00 00 00 
00 00 00 


31 31 
32 32 
33 33 
34 34 
35 35 
00 00 
00 00 
00 00 


31 31-31 31 31 31 
32 32-32 32 32 32 
33 33-33 33 33 33 
34 34-34 34 34 34 
35 35-35 35 35 35 
00 00-00 00 00 00 
00 00-00 00 00 00 
00 00-00 00 00 00 


Llevar IY a 0300H (puntero 


Р=0196 0196, Prueba de (inserción) 


-DM400 


0400 
0410 
0420 
0430 
0440 
0450 
0460 
0470 


-SY 
Y=0300 310 


4F 4E 31 
00 00 00 
00 00 00 
00 00 00 
00 00 00 
00 00 00 
00 00 00 
00 00 00 


-6193/196 


P=0196 0196’ Pase de “NUEVO” 


-nM400 


0400 
0410 
0420 
0430 
0440 
0450 
0460 
0470 


-DM400 
0400 
0410 
0420 
0430 
0440 
0450 
0460 


0470 


53 
32 
34 
33 
35 
00 
00 


00 


31 31 
00 00 
00 00 
00 00 
00 00 
00 00 
00 00 
оо 00 


Llevar ГУ а 0310H 


31 31 
32 32 
00 00 
00 00 
00 00 
00 00 
00 00 


00 00 


(Más 


31 31 
32 32 
34 34 
41 4E 


00 00 


00 00 
00 00 


00 00 


31 31-31 31 31 31 
90 00-00 00 00 00 


00 00-00 00 00 00 
00 00-00 00 00 00 


00 00-00 00 00 00 
00 00-00 00 00 00 
00 00-00 00 00 00 


00 00-00 00 00 00 


31 31-31 31 31 31 
32 32-32 32 00 00 


00 00-00 00 00 00 
00 00-00 00 00 00 


00 00-00 00 00 00 
00 00-00 00 00 00 
00 00-00 00 00 00 


00 00-00 00 00 00 


А Е . 
іпвегсіопев) 


31 31-31 31 31 31 
32 32-32 32 55 AE 
34 4D-4F AD 33 33 
54 35-35 35 35 35 


00 00-00 00 00 00 


00 00-00 00 00 00 
00 00-00 00 00 00 


00 00-00 00 00 00 


31 
32 
33 
34 
35 
00 
00 
00 


a OBJETO) 


(inserción) 


31 
43 
33 
35 


00 


00 
00 


00 


44 
00 


00 
00 


00 
00 
00 


00 


44 
34 
33 
35 


00 


00 
00 


00 


00 
00 
00 
00 
00 
00 
90 
00 


41 
34 
33 
35 
00 


00 
00 


00 


OFFFF 0168 
SIGUIE 012Ғ 
TABLEN 0102 


(siguiente OBJETO) 


44 
34 
33 
35 


00 


00 
00 


00 


Lista de objetos con 
sus posiciones en 
memoria 


S0N1111111111... 
0402222222222... 
MOM3333333333... 
UNC4444444444. .. 
АМТ5555555555... 
nca E PER 


4............. 


Configuración de la 
tabla tras el pase del 
programa 


SON1111111111,.. 


............: 
nor 
................: 
...........:ғ:ғ4:44 
............... 


................ 


............::4:4: 


Configuración de la 
tabla tras la segunda 
inserción 
SON1111111111DAD 
2222222222...... 
nm 
4............. 
eere rong 
............. 
............... 


............... 


Configuración de la 
tabla tras varias 
inserciones 
SON1111111111DAD 
2222222222UNC444 
4444444M0M333333 
3333аНт555555555 
5............ 


..4.%.............. 


4.............. 


4.............. 
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55ү 
Y=0340 320 
-6190/193 


P=0193 0193’ Pase de “BUSCA” 


El reg D indica que se ha hallado el objeto 


Contenido de los 
-OR registros 
RBC-02FF DE=FFOD HL=0340 $=0100 Р=0193 0193” CALL 0135 
А’=00 R'-0000 П’=0000 H 000 Х=0427 Y=0320 I=00 (01357) 


Dirección del objeto 


-6196/199 

р-о199 0199’ Pase de "BORRAR" (eliminación) Configuración de la 
tabla tras la 

-DM400 eliminación 


0400 53 АҒ 4E 31 31 31 31 31-31 31 31 31 31 44 41 44 50М1111111111ПАП 
0410 32 32 32 32 32 32 32 32-32 32 55 AE 43 34 34 За 22222222220МС444 
0420 34 34 34 34 34 34 34 41-4Е 54 35 35 35 35 35 35 4444444АМТ555555 
0430 35 35 35 35 41 4E 54 35-35 35 35 35 35 35 35 35 5555АМТ555555555 
0440 35 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 GS... nnn 
0450 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ..........»..... 
0460 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ................ 


0470 00 00 00 00 00 00 00 00-00 ОО 00 00 00 00 00 00 ...........»».»... 


-SY 

Y=0240 340 

тозуу Eliminar la última entrada de la tabla Nota: no hay cambio 
visible de la 

Р-0199 0199” configuración de la 
tabla 


-IMA00 
0400 53 АҒ АЕ 31 31 31 31 31-31 31 31 31 31 44 41 44 SON1111111111DAD 
0410 32 32 32 32 32 32 32 32-32 32 55 АЕ 43 34 34 34 2222222222UNC444 
0420 34 34 34 34 34 34 34 41-4Е 54 35 35 35 35 35 35 4444444АМТ555555 
0430 35 35 35 35 41 АЕ 54 35-35 35 35 35 35 35 35 25 5555$АМТ555555555 
0440 35 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 S... nm mn 
0450 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 OO ................ 
0460 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ........»....»..» 
0470 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ................ 


-0418951 Posición de memoria ТАВГЕМ; indica la verdadera longitud 
0189 03 -—— 
-6190/193 de la tabla 


F=0193 0193” Pase de “BUSCA” del objeto eliminado 
Figura 9.17 


Pase de prueba de los progra- D revela que no se ha hallado el objeto 
mas de la lista sencilla (conti- тік Це 

+ Z N =55 BC=00FF ПЕсОООП HL-0441 5=0100 F=0193 0193 CALL 0135 
nuación). А’=00 В”-0000 П”-0000 Н”-0000 Х=041А Y=0340 1=00 (01357) 


4, 


O EN 


Lista alfabética 


Esta lista o “tabla”, a diferencia de la anterior, mantiene 
todos sus elementos ordenados alfabéticamente, lo que permite 
utilizar técnicas de búsqueda más rápidas que la lineal; en este 
caso utilizaremos la búsqueda binaria. 


BUSQUEDA 


El algorítmo es el clásico de búsqueda binaria. La técnica es 
básicamente similar a la que se usa para buscar un nombre en 
una guía telefónica: se abre hacia la mitad y, según lo que 
ponga allí, se avanza o se retrocede para acercarse al nombre 
buscado; es un método rápido y bastante fácil de realizar. 

El diagrama de flujo aparece en la figura 9.18, y el programa 
en la 9.23. 

Los elementos están ordenados en la lista alfabéticamente y 
se recuperan mediante la técnica binaria o logarítmica, como 
indica el ejemplo de la figura 9.19. El procedimiento es un tanto 
complicado, porque es preciso llevar la cuenta de varias condi- 
ciones. El problema más importante es evitar la búsqueda de 
algo que no está en la lista, ya que en tal caso el programa 
escrutaría incesantemente los elementos situados justo antes y 
después del buscado; para evitarlo, se mantiene en el programa 
una bandera que conserva el valor de la de acarreo tras una 
búsqueda infructuosa. Cuando el valor INCMNT, que indica la 
magnitud en que debe incrementarse a continuación el punte- 
ro, alcanza el valor “1”, se activa otra bandera llamada 
“CERRAR” al valor de la bandera COMPR; de esta forma, 
como todos los incrementos ulteriores serán de “1”, si el 
puntero sobrepasa el punto en que debería estar el objeto, 
COMPR deja de ser igual a CERRAR, y la búsqueda termina. 
Este recurso permite, además, a la rutina NUEVO determinar la 
situación de los punteros lógicos y físicos en relación con el 
lugar al que irá el objeto. 

Por tanto, si el OBJETO buscado no está en la tabla y el 
puntero en curso se incrementa en uno, queda activada la 
bandera CERRAR. Cuando la rutina avance el paso siguiente, 
el resultado de la comparación será el contrario del anterior, las 
dos banderas dejarán de coincidir y el programa se dirigirá a la 
salida con la indicación “no hallado”. 

El otro problema importante que debe resolverse es la posi- 
bilidad de salirse de la tabla al sumar o restar el valor de 
incremento; para evitarlo se lleva a cabo una “suma” o una 
“resta” de prueba con el puntero lógico y el valor de longitud 
que registra el número real de elementos, no las posiciones 
fisicas de memoria utilizadas por los punteros físicos. 
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Figura 9.18 
Diagrama de flujo de la bús- 
queda binaria. 


BANDERAS = 0 


SEÑALAR LA BASE DE LA TABLA 


POSICION LOGICA = 
VALOR DE INCREMENTO = 
LONGITUD DE LA TABLA/2 
(SUMAR 1 SI ES IMPAR) 


SEÑALAR El CENTRO DE LA TABLA 


'ALOR DE INCREMENTO = VALOR DE INCREMENTO/2 


SUMAR 1 $1 ES IMPAR 
COMPARAR OBJETO CON ENTRADA 


GUARDAR EL ACARREO (SIGNO DE 
COMPARACION) EN LA BANDERA COMPR 


¿VALE 1 EL 
INCREMENTO? 


SI 
NO HALLADO 
д NO 


(ENTRADA) 


HALLADA 


(SIGUIENTE PRUEBA) (ULTIMA) 


Pun 


(SIGUIENTE PRUEBA) (ULTIMA) 


5! 
¿CERRAR = 0? 


CERRAR = 
COMPR 


¿CERRAR = 
ADA + 
МО HALLAD, COMPR? 
RESTAR FF SALTO SUMAR 1 


A COMPR 


¿SE SALDRA EL $1 
INCREMENTO DE 


LA TABLA? 


5! 


¿FINAL DE 


LA TABLA? m 


HALLADA 


NO 
NO 
(DEALTO) 
ACTUALIZAR P PUNIE 
CTUALIZAR LOS PUNTEROS 
(ENTRAD) 


51 ¿SE SALDRA ЕЁ 
INCREMENTO DE 


LA TABLA? 


NO 
ACTUALIZAR 
PUNTEROS 


(ENTRAD) 


HALLADA 


(DEBAJO) 


INCREMENTO = 1 
LOS PUNTEROS ARES 
Figura 9.18 


Diagrama de flujo de la bús- 
queda binaria (continuación). (ENTRAD) 
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Figura 9.19 
Búsqueda binaria. 
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(0121) LD A, C 
A 


(NO) (NO) 


TES 


(2) XYZ 
PRIMER INTENTO SEGUNDO INTENTO 
INTERVALO DE BUSQUEDA = 5 INTERVALO DE BUSQUEDA = 2 


En resumidas cuentas, el programa emplea dos banderas 
para memorizar la información: COMPR y CERRAR. La pri- 
mera mantiene el valor “0” ó “1” del acarreo tras la última 
comparación, lo que determina si el elemento en prueba es 
mayor о menor que el verificado inmediatamente antes; С 
señala la relación: si es “1”, quiere decir que el elemento es 
menor que el objeto, y COMPR se pone a “1”; si es “0”, 
significa que el elemento es mayor que el objeto, y COMPR se 
pone a “FF”. 

La bandera CERRAR se hace igual a COMPR cuando el 
incremento de búsqueda INCMNT alcanza el valor “1”; $1 al 
paso siguiente COMPR y CERRAR dejan de ser iguales, es que 
el elemento buscado no se encuentra. 

El programa utiliza también las variables siguientes: 


LOGPOS indica la posición lógica en la tabla (número 
de elemento). 

INCMNT representa el valor en que hay que incremen- 
tar о decrementar el puntero en curso si 
falla la comparación siguiente. 


LONTAB, como es habitual, representa la longitud total 
de la lista. 


LOGPOS e INCMNT se comparan con TABLA, para ga- 
rantizar que no se sobrepasan los límites de la lista. 

El programa, llamado “BUSQ”, aparece en la figura 9.23, y 
reside en las posiciones de memoria 010А a 0109. Debe estu- 
diarse con atención, porque es mucho más complicado que el 
de búsqueda lineal. 

Como el intervalo de búsqueda puede ser par o impar, es 
preciso introducir una corrección (en efecto, el programa no 
puede señalar el elemento central de una lista de cuatro); si es 
impar, se emplea un “truco” muy sencillo para dirigir el puntero 
hacia el elemento central: la división por 2 va acompañada de un 
desplazamiento a la derecha, que se hace sumando a dicho 
puntero el “1”, el cual pasa al acarreo tras aplicar la instrucción 
SRL a un intervalo impar. 

A continuación se compara OBJETO con el elemento cen- 
tral del nuevo intervalo de búsqueda; si la comparación es 
positiva, el programa termina. En caso contrario (^NOBUEN") 
se reinicia a "0" el acarreo para indicar que OBJETO es menor 
que el elemento. Cuando INCMNT alcanza el valor “1”, se 
verifica la bandera CERRAR, que se había inicializado a “0”, 
para ver si está activada, y se activa en caso negativo; si ya está 
activada, se procede a una verificación para determinar si se 
ha pasado la posición en que debería estar OBJETO sin encon- 
trarlo. 

Cuando el acarreo vale “1”, el puntero señala el elemento 
situado por debajo de OBJETO. 


INSERCION DE UN ELEMENTO 


Para insertar un nuevo elemento se lleva a cabo una bús- 
queda binaria. Si ya se encuentra en la tabla, es que no hay que 
introducirlo (supondremos que todos los elementos son distin- 
tos). Si no aparece, se inserta inmediatamente antes o después 
del último elemento comparado, según lo indicado por la ban- 
dera COMPR. Todos los elementos que siguen al nuevo deben 
descender la posición de un bloque para hacerle sitio. 

El método de inserción se ilustra en la figura 9.20 y el 
correspondiente programa aparece en la 9.23, Se llama NUEVO, 
y empieza en la posición de memoria 01DA. Obsérvese que 
se han empleado las instrucciones automáticas del 780 LDDR 
y LDIR para hacer transferencias de bloques eficaces. 
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| 


ANTES DESPUES 


BASETA—* 


NUEVO 
ELEMENTO | 


Еїдига 9.20 OBJETO BAC 
Inserción de “ВАС” 


ELIMINACION DE UN ELEMENTO 


Como antes, se empieza por una búsqueda binaria para 
encontrar el objeto. Si falla, es que no se encuentra en la lista y 
no puede eliminarse. Si aparece, se elimina, y todos los que le 
siguen ascienden una posición, como se ve en el ejemplo de la 
figura 9.21. El programa aparece en la 9.23, y el diagrama de 
flujo, en la 9.22. Se llama “BORRAR” y reside en la dirección 
0221. 

La figura 9.24 recoge un pase de prueba de todos los pro- у 
gramas propuestos. 


ANTES DESPUES 


SUBIR 


Figura 9.21 
Eliminación de “BAC”, BORRAR 
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| 


Figura 9.22 
Diagrama de flujo de elimina- 
ción (lista alfabética). 


BORRAR 


| 


мо 
SALIDA 


5! 


CONTAR LOS ELEMENTOS 
QUE SIGUEN AL QUE 
VA A BORRARSE 


RESULTADO = CONTADOR 
(LOGPOS) 


SEÑALAR ENTRADA SIGUIENTE 
PUNTERO = TEMP (FUENTE) 


TRANSFERIRLO UN BLOQUE 
HACIA ARRIBA 


SEÑALAR LA ENTRADA SIGUIENTE 
PUNTERO = PUNTERO (DESTINO) 


DECREMENTAR LOGPOS 


(BAJOTA) 
ACTIVAR LAS DOS BANDERAS 


RTS 
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Figura 9.23 
Programa de búsqueda binaria. 


540 


0100 
0100 
0102 
0104 
0106 
0108 


010A 
010C 
010F 
0112 
0115 
0116 
0119 
011B 
0110 
O11E 


O11F 
0122 
0123 
0124 
0127 
0128 
0129 
012B 


012С 
012Е 
0150 
0151 
0154 
0157 
015А 
0150 
0140 
0145 
0146 
0149 
014С 


014E 
0151 
0153 
0156 
0157 
0158 
0158 
015Е 
015Ғ 
0162 
0163 
0166 
0167 
о16а 
0160 


0170 


0175 
0175 
0176 
0177 
017А 


0170 
017Е 
0181 
0182 


0185 
0186 
0189 


018A 
018C 


5402 
5502 
5602 
5702 
5902 


SE00 
320001 
320201 


3A0401 
CB3F 
СЕОО 
4F 

47 


CAC401 
SF 

1D 
CDC701 
19 

ES 
DDE1 
79 


CB3F 
СЕОО 
АҒ 
DD7EOO 
FDBEOO 
C24CO1 
DD7EO1 
FDBEO1 
C24CO1 
DD7EO2 
FDBEO2 
CAC&01 
3E01 


рА5301 


5А0001 
А7 
CAEDO1 
57 
3A0201 
92 
C27301 
С5С401 
$10201 


$20001 


CDC701 
340201 


3c 
C2A001 
78 
91 


САВҒО1 
DABFO1 
47 


EDS2 
C32801 


CERRAR 
COMPR 

LONTAB 
BASETA 
LONENT 


; 
BUSQ 


ENTRAD 


NOBUEN 
; 


ТЕЗТЗ 


NOCERR 


SIGTES 


`. 


#0100 
FIN 

FIN+1 
FIN+2 
FIN+3 
FIN+5 


А, 0 
(CERRAR) , А 
(COMPR) , А 
о, А 

HL, (BASETA) 
A, (LONTAB) 


A, (1Х+0) 
(1Y+0) 
NZ, NOBUEN 
A, CIX*1) 
(уж) 
№7, NOBUEN 
А, (1Х+2) 


с, TESTS 
А, HOFF 
(COMPR) , A 
A,C 

A 

NZ, SIGTES 
A, (CERRAR) 
A 

Z, NOCERR 
D, A 

A, (COMPR) 
D 
NZ,SIGTES 
NOVALE 

A, (COMPR) 


(CERRAR),A 


IX 
HL 

E,C 

MULT 

A, (COMPR) 


A 
NZ, SUMALO 
A,B 

с 


7, DEBAJO 
C, DEBAJO 
B,A 


HL, DE 
ENTRAD 


¿POSICIONES DE FLAG CERO 


; INICIALIZA HL 
;DIVIDE POR 2 


5 ALMACENA VALOR INCREMENTO 
¿ALMACENA VALOR DE 
¿POSICION LOGICA 
¿COMPRUEBA SI LOGITUD ES O 
¿MULTIPLICA (E-1)*LONENT 


¿COLOCA HL EN MITAD TABLA 


¿CARGA HL EN IX 


3DIVIDE EL VALOR DE 
3 INCREMENTO POR 2 


3COMPARA LA PRIMERA LETRA 
¿COMPARA LA SEGUNDA LETRA 


3 COMPARA LA TERCERA LETRA 


¿FIJA FLAG DE RESULTADO DE 
¿LA COMPARACION AL 
3..RESULTADO DE ELLA (1,FF) 


¡ES 1 VALOR DE INCREMENTO ? 


¡FIJA FLAG DE CIERRE A LA 
¿DIRECCION DE 

5 BUSQUEDA PARA PREVENIR UNA 
3 REPETICION 

¿PREPARA HL Y DE PARA SUMAR 
30 RESTAR EL VALOR DE INCR. 


3 COMPRUEBA SI QUIERE SUMAR 
30 RESTAR 


3 COMPRUEBA SI AL RESTAR 
35 TENDRA UN VALOR DEMASIADO 
¿BAJO 


¿FIJA EL VALOR DE LA NUEVA 
¿POSICION LOGICA 
¿CAMBIA LA MISMA DIRECCION 


Figura 9.23 
Programa de búsqueda binaria 
(continuación). 


0182 
0190 
0191 
0194 
0198 
0199 
0194 
019С 
0190 
0140 
0185 
0184 
01А5 
о1ав 


01А9 


O1AA 
O1AB 
O1AC 
O1AF 


01В0 
0185 
0187 
01B8 
01B9 
O1BB 
O1BE 
01С1 
о1с4 
о1се 


0107 
о1св 


O1C9 
O1CB 
O1CE 
0102 
0105 
0104 
0106 
0107 
0108 
01р9 


O1DA 
0100 
O1DE 
O1E1 
01Е4 
01Е5 
O1E8 
O1EB 
O1EC 
O1EF 
O1FS 
O1F4 
O1F7 
O1F8 


O1FB 
O1FC 
O1FF 


0200 
0203 
0204 
0205 
0206 
0209 
020А 
0208 


78 

3D 
CAC401 
Ер5В0801 


05 


91 
СААРО1 
19 


78 


C32801 
81 


CAC401 
Е0580801 
19 

04 

OEO1 
540201 
520001 
C32801 
16FF 

c9 


ES 
cS 


1600 
210000 
ED4BOBO1 
41 

19 


CAF701 
Е0580801 
19 
СЗЕВО: 


90 
СА1602 
SF 


CDC701 
19 

2B 

EB 
240801 
19 

EB 
Ер4В0801 


770 
780 
790 
800 
810 
820 
830 
840 
850 
850 
870 
880 
890 
900 
910 
920 
930 
940 
950 
960 
970 
980 
990 
1000 
1010 
1020 
1030 
1040 
1050 
1060 
1070 
1080 
1090 
1100 
1110 
1120 
1130 
1140 
1150 
1160 
1170 
1180 
1190 
1200 
1210 
1220 
1230 
1240 
1250 
1260 
1270 
1280 
1290 
1300 
1310 
1320 
1330 
1340 
1350 
1560 
1570 
1380 
1390 
1400 
1410 
1420 
1430 
1440 
1450 
1460 
1470 
1480 
1490 
1500 
1510 
1520 


DEBAJO LD 
DEC 


SUMALO LD 


- 


DEALTO ADD 
5 


CREAL LD 


NOVALE LD 
VALE RET 


3 
MULT PUSH 


3 


SUMAS ADD 


Zw ww 


UEVO CALL 


LADOAL DEC 
SISTEM LD 
5 


- 


MOVIM LD 


A,B 

A 

2,NOVALE 
DE, (LONENT) 
HL,DE 

B 

CREAL 


^, (LONTAB) 
B 

с 

с, DEALTO 
HL, DE 


а, В 


2,NOVALE 
DE, (LONENT) 
HL, DE 

B 

С,1 

A, (COMPR) 
(CERRAR) , À 
ENTRAD 

D, OFF 


HL 
BC 


D,0 
HL, 0000 

BC, (LONENT) 
B,C 

HL, DE 
SUMAS 

BC 

DE, HL 

HL 


BUSQ 

D 
NZ,SALIR 
^, (LONTAB) 
^ 

Z, INSERT 
A, (COMPR) 
A 
Z,LADOAL 
DE, (LONENT) 
HL,DE 
SISTEM 

B 

A, (LONTAB) 


B 
Z, INSERT 
Е, А 


MULT 

HL, DE 

HL 

DE,HL 

HL, (LONENT) 
HL,DE 
DE,HL 

BC, (LONENT) 


¿VE SI LA POSICION ES UNO 


3SI ES ASI, SE VA 
¿SOLO RESTA UNA POSICION 


¿CAMBIA LA POSICION LOGICA 


3 COMPRUEBA SI LA POSICION 
¿ACTUAL MAS EL INCREMENTO 
¿SON MAYORES QUE LONTAB 


3SI NO ES ASI CAMBIA LA 
¿DIRECCION ACTUAL 
¿CAMBIA EL VALOR DE LA 
¿POSICION LOGICA 


¿VE SI LA POSICION ES AL 
¿PRINCIPIO DE LA 

¿TABLA (IGUAL A LONTAB-B) 
3 SUMA UNA POSICION ENTRADA 


3 INCREMENTA POSICION LOGICA 
3FIJA A UNO EL INCREMENTO 
3DEFINE FLAG DE CIERRE 
РАВА COMPARAR RESULTADO 


¿MULTIPLICA E POR EL VALOR 
3DE (LONENT) COLOCANDOLO 
ЗЕМ DE 


3VE S1 OBJETO YA ESTA ALLI 


; COMPR=1, FIJA HL A DONDE 
3DEBE IR EL OBJETO 


3COMPR-O, B PARA RESTAR 
3VE CUANTAS ENTRADAS SE НАМ 
3 HECHO 


¿FIJA HL А LA ULTIMA 
¿POSICION DE LA ULTIMA 
¿ENTRADA 


;DE ES UNA ENTRADA ANTES HL 


¿SHIFT UNA ENTRADA DE MEM. 
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Figura 9.23 
Programa de búsqueda binaria 
(continuación). 


O20F EDB8 1530 LDDR 
0211 3D 1540 DEC A 
0212 C20B02 1550 JP NZ, МОУІМ ¿REPITE SI ES NECESARIO 
0215 23 1560 INC HL 
0216 FDES 1570 INSERT PUSH IY 3 CARGA EN ESPACIO VACIO 
0218 D1 1580 POP DE 
0219 EB 1590 EX DE,HL 
0214 Е0480801 1600 LD BC, (LONENT) 
021Е EDBO 1510 LDIR 
0220 3A0401 1520 LD A, (LONTAB) 3 INCREMENTA LONGITUD TABLA 
0223 SC 1630 INC а 
0224 320401 1640 LD (LONTAB),A 
0227 O1FFFF 1650 LD ВС, HOFFFF ¿MUESTRA LO QUE HA HECHO 
022А C? 1660 SALIR ВЕТ 
1670 ; 
1680 ; 
1690 ; 
022В CDOAO! 1700 BORRAR CALL BUSQ 3COGE LA DIRECCION OBJETO 
022Е 14 1710 INC D ¿VE SI OBJETO ESTA ALLI 
022F СА5502 1720 JP 2, SALIRE 
0232 EDSBOBO1 1730 LD DE, (LONENT) 
0256 EB 1740 EX DE, НЫ 
0237 19 1750 ADD HL,DE 3DE ES POS. DE OBJETO, HL 
0238 540401 1760 LD A, (LONTAB) ¿ESTA UNA ENTRADA ANTES 
023B 90 1770 SUB B 3VE CUANTAS ENTRADAS SE HAN 
023C CA4902 1780 JP 7, BAJOTA 3 HECHO 
O23F Е0480801 1790 SHIFT LD BC, (LONENT) 
0243 EDBO 1800 LDIR ¿SHIFT ABAJO UNA LONENT 
0245 3D 1810 DEC А 
0246 C23F02 1820 JP NZ, SHIFT 
0249 3A0401 1830 BAJOTA LD A, (LONTAB) ; DECREMENTA LONGITUD TABLA 
024C 3D 1840 DEC А 
024D 320401 1850 LD (LONTAB),A 
0250 O1FFFF 1860 LD ВС, #OFFFF ¿MUESTRA LO QUE HA REALIZ. 
0253 C9 1870 SALIRE RET 
1880 ; 
0254 1890 FIN END 
BAJOTA 0249 BASETA 0105 MULT 01С7 NOBUEN 014С 
BORRAR 022В BUSQ 010A NOCERR 0160 NOVALE O1C4 
CERRAR 0100 COMPR 0102 NUEVO 0104 SALIR 022A 
CREAL  O1B9 DEALTO O1AF SALIRE 0253 SHIFT 023F 
DEBAJO 018F ENTRAD 0128 SIGTES 0173 SISTEM O1F8 
FIN 0254 INSERT 0216 SUMALO O1A0 SUMAS 010$ 
LADOAL O1F7 LONENT 0108 TESTS 0155 VALE O1C& 


LONTAB 0104 MOVIM 0208 


Lista encadenada 


Los elementos de esta lista contienen una etiqueta de tres 
caracteres alfanuméricos, 250 bytes de datos, un apuntador de 
dos bytes con la dirección de partida del elemento siguiente y 
un marcador de un byte; cuando éste vale "1", impide a la 
rutina de inserción colocar un elemento nuevo en el lugar de 
otro preexistente. 

Además, para facilitar la recuperación, hay un directorio que 
contiene un apuntador orientado hacia la primera entrada de 
cada letra del alfabeto. Las etiquetas son caracteres alfabéticos 
ASCII. Todos los apuntadores del final de la lista tienen un 
valor NIL, que en este caso se ha elegido igual a la base de la 
tabla, ya que dicho valor nunca debe aparecer en el interior de 
la lista encadenada. 


Figura 9.24 


Tabla inicial 


1м400 
0400 00 00 00 00 00 00 00 00-00 00 00 00 00 00 0000 . vos 
0410 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 . sos. 
0420 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 . 0 
0430 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 . .-.. 
0440 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00, o ooo 
0450 00 00 00 00 00 00 00 00-00 00 00 00 00 00 0000 . n 
9460 00 00 00 00 00 00 00 00-00 00 00 00 00 00 OO OO ................ 
0470 00 00 00 00 00 00 00 00-00 оо 00 оо OO 00 OO OO ................ 
Relación de objetos y 
sus posiciones en 
memoria 
-M300 
0300 53 АҒ АҒ 31 31 31 31 31-31 31 31 31 31 00 00 00 50М1111111111... 
0310 44 41 44 32 32 32 32 32-32 32 32 32 32 00 00 00 HAN? 222... 
0320 АП АҒ АП 33 33 33 33 33-33 33 33 33 33 00 00 00  MOM333333 33... 
0330 55 AF 43 34 34 34 34 34-34 34 34 44 за 00 00 00 14444444444... 
0340 41 4Е 54 35 35 39 35 35 35 35 35 35 35 00 00 00 ANISSSSSSSSS, а 
0350 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ................ 
0360 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ................ 
0370 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ЕС 
-SY 
У=0000 320 
263/2066 Pase de “NUEVO” 
F-06066 0266“ 
пмаоо Tabla tras la inserción 
0400 аП AF 40 33 33 33 33 33-33 33 33 33 45 00 00 00 MOM3333333333,.,. 
0410 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ................ 
0420 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 OO »...»........... 
9430 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 dorar t! n n 
0440 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 eret t] 
0450 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ................ 
0450 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 sers 
0470 00 00 00 00 00 оо 00 00-00 00 00 00 00 00 00 ОО .........,....... 
SY 
У=03 
и Pase de “NUEVO” con otro bojeto 
10266 0266’ Tabla tras la inserción. 
Nota: la tabla sigue 
siendo alfabética 
-HM400 
0400 44 41 44 32 32 32 32 32-32 32 32 32 32 АП 4 ан 22222M0M 
0410 33 33 33 33 33 33 33 33-33 33 00 00 00 00 00 00 vs. 
0420 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ett tn ng 
0430 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 n 
0440 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 ОО ................ 
0450 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 OO ........»...... 
0440 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 OO ................ 
оао 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 8 бб б 4 ча 
* + «(inserciones adicionales) * * • 
Configuración de la 
tabla tras insertar 
-0м400 todos los objetos 
0400 41 4E 54 35 35 35 35 35-35 35 35 35 35 44 41 44 АМТ5555555555ПАП 
0410 32 32 32 32 32 32 32 32-32 32 АП АҒ АП 33 33 33 2222222222M0M333 
0420 33 33 33 33 33 33 33 53-АҒ 4E 31 31 31 31 31 31  3333333S0N111111 
0430 31 31 31 31 55 4E 43 34-34 34 34 34 34 34 34 За 11110МС444444444 
0440 34 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 4............... 
0450 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ................ 
0460 00 00 00 00 00 00 00 00-00 00 00 оо оо 00 оо 00 ................ 
0470 00 00 00 00 00 00 00 00-00 00 00 оо 00 00 оо 00 ................ 
-SY 
Y=0340 300 
-6260/263 


Р=0263 0263” 


~IR 


4'-00 B'-0000 D'-20000 Н’=0000 X=0427 Y=0300 I-00 


1— Hallado 
Z N А=4Е ВС=0401 1Е-0000 HL-0427 S=0100 F=0263 0263, CALL 0100 


Pase de "BUSQ" de 'SON' (en la dirección 0300) 


(0107) 


Pase de prueba de la lista alfa- Dirección del objeto en la tabla 


bética. (comprobar en la tabla de arriba que está SON?) 


-6266/269 


Pase de “BORRAR” ‘SON’ Configuración de la 
P=0269 0269” tabla tras la 
eliminación. Nota: 
UNC se ha 
desplazado hacia 
arriba; la última 
entrada UNC se 


-рмаоо ignora 

0400 41 АЕ 54 35 35 35 35 35-35 35 35 35 35 44 41 44 ANT5555555555DAD 
0410 32 32 32 32 32 32 32 32-32 32 АП AF AD 33 33 33 2222222222M0M333 
0420 33 33 33 33 33 35 33 55-4Е 43 34 За 34 За 34 34 3333333UNC444444 
0430 34 34 34 34 55 AE 43 34-34 34 За За За 34 34 за 4444UNC444444444 
0440 За оо 00 00 00 00 00 00-00 00 00 00 00 00 00 00 4............... 
0450 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ................ 
0460 00 00 00 00 00 00 00 00-00 00 оо 00 оо 00 00 00 ................ 
0470 00 00 00 00 00 00 00 00-00 00 00 00 оо 00 00 00 ................ 


-6260/263 


Nuevo pase de “BUSQ” de ‘SON? 

P=0263 0263' 

ET. No hallado 

S N  A-FE BC-0401 DE-FFOD HL-0427 S-0100 P-0263 0263' CALL 0110 
А’=00 В”-0000 D'-0000 H'-0000 Х-0427 Y=0300 100 (01D0*) 


-G263/266 
Reinsertar el objeto (SON”) 
P=0266 0266* 
Configuración actual 
de la tabla. 
Compárese con la 
anterior a “BORRAR” 


0400 41 АЕ 54 35 35 35 35 35-35 35 35 35 35 44 41 44 АМТ5555555555ПАП 
0410 32 32 32 32 32 32 32 32-32 32 АП АҒ АП 33 33 33 2222222222М0М333 
0420 33 33 33 33 33 33 33 53-АҒ AE 31 31 31 31 31 31 3332333650М111111 
0430 31 31 31 31 55 АЕ 43 34-34 34 34 34 34 34 34 34 11111/МС444444444 
0440 34 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 4....».....»».... 
0450 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 .....»..»...».».. 
0460 00 оо 00 оо 00 00 00 00-00 00 00 00 00 00 00 00 ................ 
0470 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ................ 


Figura 9.24 Revela que se ha ejecutado la acción 


А -OR 
Pase de prueba de la lista alfa- A=05 BC-FFFF DE=0434 HL=030N S=0100 Р=0266 0266’ CALL 0221 


bética (continuación). А’=00 В’=0000 D'/20000 Н”-0000 X=0427 Y-0300 1-00 (02217) 


pulaciones obvias de los apuntadores. Utilizan la bandera 
INDEXA para indicar si el que señala un objeto procede de un 
elemento anterior de la lista o del directorio. Los programas 
correspondientes se encuentran en la figura 9.29. Las estructuras 
de datos aparecen en la 9.25. 


DIRECTORIO 


PUNTERO 


PUNTERO 


Figura 9.25 

Estructura de la lista encade- 
nada. 
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Los programas de inserción y eliminación ejecutan las mani- 


Una aplicación de esta estructura de datos sería un libro de 
direcciones informatizado, en el que cada persona estuviese 
representada por un código único de tres letras (sus iniciales, 
por ejemplo) y tuviese en el campo de datos una dirección 
simplificada y el número de teléfono (hasta 250 caracteres). 
Examinemos dicha estructura más de cerca. El formato de cada 
entrada es: 


ЕЕ ИЕ ЕЕ 


Ne a ` a х= me 


etiqueta única datos (1 a 250 bytes) puntero 


(ASCII) al siguiente 
"m ocupado 


Las convenciones son las habituales: 


LONENT: longitud total en bytes. 
BASETA: dirección de la base de la lista. 


Se supone que la dirección de OBJETO reside siempre en el 
registro IY antes de entrar al programa. REFBAS señala la 
dirección de la base del directorio o "tabla de referencia". 

Cada una de las direcciones de dos bytes de éste apunta a la 
primera aparición de la letra a que corresponde en la lista, de 
manera que cada uno de los grupos de elementos que compar- 
ten la misma inicial en sus etiquetas forman, en realidad, una 
lista independiente dentro de la estructura general. Esta caracte- 
rística facilita la búsqueda, y es análoga a una agenda de 
direcciones. Obsérvese que durante la inserción y la eliminación 
no se mueve ningún dato, sino que únicamente se modifican los 
punteros, como en cualquier estructura de lista encadenada. 

Si no aparece ninguna entrada con una inicial específica o si 
no hay ninguna que siga alfabéticamente a otra preexistente, sus 
punteros señalan al comienzo de la tabla (— "NIL"). Al 
fondo de ésta se conviene en almacenar un valor tal que el 
valor absoluto de la diferencia entre él y *Z" sea mayor que la 
diferencia entre “A” y “Z”; esto constituye el indicador de fin de 
tabla (EOT). En este caso, el EOT ocupa la misma cantidad de 
memoria que un elemento normal, pero si se desea puede tener 
un solo byte; las letras son caracteres alfabéticos en código 
ASCII. Si se modifica este extremo, habría que cambiar también 
la constante de la rutina PRETAB. 

El marcador de fin de tabla se lleva al valor del principio de 
la misma (“NIL”). 
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Figura 9.26 
Búsqueda en la lista encade- 
nada. 
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Se conviene en llevar los “punteros NIL" del final de una 
serie, o de una posición del directorio que no señale una serie, 
al valor de la base de la tabla, para disponer de una identifica- 
ción ünica, aunque podría emplearse otra convención. En con- 
creto, si se emplea un marcador diferente como EOT se ahorra 
algo de espacio, porque no es preciso mantener entradas NIL 
para los artículos inexistentes. 

La inserción y la eliminación se llevan a cabo de la forma 
habitual (véase la parte I de este mismo capítulo) simplemente 
modificando los punteros adecuados. Se usa la bandera 
INDEXA para indicar si el puntero que señala el objeto está 
en la tabla de referencia o en otro elemento. 


BUSQUEDA 


El programa de büsqueda (BUSQ) reside en las posiciones 
de memoria 0108 a 015D y utiliza la subrutina PRETAB de la 
dirección 01D9. 

El principio de büsqueda es bastante obvio: 


1. Se obtiene en el directorio la entrada correspondiente a la 
letra del alfabeto situada en la primera posición de la 
etiqueta de objeto. 


2. Se obtiene el puntero y se accede al elemento. Si es NIL, 
la entrada no existe. 


3. En caso contrario, se compara el elemento con OBJETO; 
si coinciden, la búsqueda ha concluido positivamente. Si 
no, el apuntador se orienta hacia la entrada inmediata- 
mente inferior. 


4. Se vuelve a 2. 


La figura 9.26 recoge un ejemplo. 


A-PUNTERO 


OBJETO AZC 


(HALLADO) 


(HAY QUE DAR 4 PASOS) 


| 
| 
| 


— 


Figura 9.27 
Ejemplo de inserción en la lista 


encadenada. 


INSERCION 


Es básicamente una búsqueda seguida de una inserción 
cuando se encuentra un elemento “NIL”. 

Se sitúa un bloque de almacenamiento, para la nueva incor- 
poración, a continuación del marcador ЕОТ, buscando para 
ello un indicador de ocupación en posición “disponible”. 

El programa aparece en la figura 9.29, se llama “NUEVO” y 
reside en las direcciones 015Е a О1АВ. La figura 9.27 recoge un 
ejemplo. 


A-PUNTERO 
B-PUNTERO 
C-PUNTERO 


B-PUNTERO 
C-PUNTERO 


cas OBJETO 


DESPUES 


ELIMINACION 


Para eliminar un elemento se sitüa su indicador de ocupa- 
ción en posición “disponible” y se ajusta el puntero al mismo 
desde el directorio o desde el elemento anterior. 

El programa se llama “BORRAR”, y reside en las direcciones 
O1AC a 0108. La figura 9.28 recoge un ejemplo de eliminación. 
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Figura 9.28 
Ejemplo de eliminación. 


(ANTES) 


ono» 


DAF PUNTERO 


DOC PUNTERO 


ELIMINAR 


(DESPUES) 


ona» 


DOC PUNTERO 


OBSERVE QUE DAF NO SE BORRA, SINO QUE SE DEJA “INVISIBLE” 


г------- 


| DAF ! 


+ 


Figura 9.29 
Programas de la lista encade- 
nada. 


0100 
0100 
0102 
0104 
0106 


0108 
010A 
010В 
010C 
010Е 


0112 


0113 
0114 
0115 
0116 
0117 
0118 
011A 


0110 
O11F 
0122 
0125 
0128 
012B 
012Е 
0131 
0134 
0157 
013A 
0150 
0140 
0145 
0146 
0148 
0149 
O14C 
014D 


O14E 


EEO1 
EFO1 
F101 
F301 


3E00 
47 

эс 
520001 
CDD901 


1A 


FE7C 
025001 
DD7EOO 
FDBEOO 
DA4601 
C25D01 
DD7EO1 
FDBEO1 
DA4501 
C25D01 
DD7E02 
DDBEO2 
CASBO1 
025001 
DDES 
01 
2А0601 
19 

4Е 


23 


O14F 46 


0159” 


0151 
0155 
0155 
0158 
015B 
0150 


015Е 
0161 
0162 
0165 


0166 
0169 


016A 
0160 


O16E 
О16Е 
0170 
0171 
0172 
0175 


с5 
DDE1 
3E00 
320001 
C31401 
O6FF 
C9 


спово1 
04 
сааво1 
05 
280201 
ЕВ 


2840601 


СА6901 


10 

20 

30 

40 

50 

фо 

70 

во 

90 
100 
110 
120 
150 
140 
150 
160 
170 
180 
190 
200 
210 
220 
230 
240 
250 
260 
270 
280 
290 
300 
310 
320 
330 
340 
350 
360 
370 
380 
390 
400 
410 
420 
430 
440 
450 
460 
470 
480 
490 
500 
$10 
520 
550 
540 
550 
560 
570 
580 
590 
600 
610 
620 
630 
640 
650 
660 
670 
680 
690 
700 
710 
720 
730 


INDEXA 
BASETA 
REFBAS 
LONENT 


j 
BUSQ 


COMPAR 


E 


NOVALE 


ENCONT 
NOENCO 


Н 
; 
Н 
N 


UEVO 


3 


, 
SIGUIE 
3 


#0100 
РІМ 

FIN+1 
FIN+3 
FIN*S 


A,0 
B,A 

A 
(INDEXA),A 
PRETAB 


A, (DE) 


L,A 
DE 

A, (DE) 
H,A 

HL 

IX 

A, (1X+0) 


#7с 
NC, NOENCO 
A, (1Х+0) 
(1Y+0) 
C,NOVALE 
NZ, NOENCO 
А, (1Х+1) 
OY) 
C,NOVALE 
NZ, NOENCO 
A, (1Х+2) 
(1у+2) 

Z, ENCONT 
МС, NOENCO 
IX 

DE 

HL, (LONENT) 
HL,DE 

C, (HL) 


HL 
B, (HL) 

BC 

IX 

4,0 
(INDEXA),A 
COMPAR 

В, НОЕЕ 


BUSQ 
B 
Z,SALIR 
DE 


HL, (BASETA) 
DE, HL 


HL, (LONENT) 
HL 


HL 
HL 
HL,DE 
A, (HL) 
^ 


Z,SIGUIE 


3 INICIALIZA FLAGS 


3COGE LA DIRECCION DEL 
¿PUNTERO DE INDICE 
¿COLOCA EL CONTENIDO DEL 
¿PUNTERO EN HL 


¿MIRA LA PRIMERA LETRA DE 
3 ENTRADA 
¿VE SI ES MARCADOR EOT 


3 COMPARA LAS PRIMERAS LETR. 
3 COMPARA SEGUNDAS LETRAS 


3 COMPARA TERCERAS LETRAS 


¿SALTA AL PUNTERO DE ENTR. 


3PONE EL VALOR DEL PUNTERO 
ЗЕМ BC 


¿CARGA IX СОМ EL PUNTERO 


3RESET DE FLAG 


3VE DONDE DEBE IR EL OBJETO 


3ALMACENA DIRECCION PREVIA 
3DE ENTRADA 

¿BUSCA ESPACIO EN LA TABLA 
SPARA ENTRADAS NUEVAS 
3MUEVE AL FINAL DE LA 
¿ENTRADA SIGUIENTE 


; ЗУМА TRES PARA LA LONGITUD 
3DE ENTRADA REAL 


$51 HAY ALGO ALLI, LO 
3 INTENTA OTRA VEZ 
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Figura 9.29 
Programas de la lista encade- 
nada (continuación). 


550 


0176 
0177 


0178 
017A 
017B 


017F 
0181 


0183 


0184 
0185 
0186 
0187 
0188 
0189 


018B 


018C 
9018F 
0190 
0193 
0194 
0198 


0199 
019A 


019B 
019С 
0190 
O1A0 
01А1 


0144 
0185 
0186 
0147 
01858 
O1AB 


O1AC 


O1AF 
01В0 
0185 


0185 
O1B6 
O1BA 
O1BB 
O1BC 
O1BD 
O1BE 
O1BF 
O1C1 


01C4 
O1CS 
о1с8 
O1CB 
O1CC 
O1CF 


0102 
0103 
0104 


El 
Ер4В0601 


EDBO 
DDES 


El 


EB 
73 
25 
72 
25 
$601 


El 
3A0001 
3D 
Е0580601 
19 

01 


75 


C34801 
Сі 
CDD901 


O1FFFF 
с? 


CDO801 


04 
C2D801 
DDES 


El 
ED4BO601 


C3D301 
200501 


19 
71 
70 


“e 


“. 


SETINX 


TERMIN 
SALIR 


$ 
; 
; 
BORRAR 
Н 


САМВІА 


3 


MOVIN 


DE 
DE 


IY 
HL 
BC, (LONENT) 


A, (INDEXA) 
A 

2, SETINX 
(SP), HL 

DE, (LONENT) 
HL, DE 


DE 
(HL),E 


HL 
(HL?) , D 
TERMIN 
BC 
PRETAB 


DE, HL 
(HL), E 

HL 

(HL), D 
ВС, #ОЕЕЕЕ 


BUSQ 


B 
NZ, SALIRE 
IX 


HL 
BC, (LONENT) 
HL, BC 

C, (HL) 

HL 

B, (HL) 

HL 

(HL),0 

^, (INDEXA) 


A 
NZ, CAMBIA 
PRETAB 

DE, HL 

MOVIN 

HL, (LONENT) 


HL, DE 
(HL) „с 
(HL),B 


3GUARDA LA POSICION DEL 
¿ESPACIO VACIO 
¿MUEVE IY A HL 


¿COLOCA EL OBJETO EN LA 
3 TABLA 


¿PONE LA DIRECCION DE 
¿ENTRADA DETRAS DEL OBJETO 
j.e. EN LA POSICION DEL 

3 PUNTERO 


3DEFINE EL MARCADOR DE 
3 ОСУРАСТОМ 

3COGE LA DIRECCION DONDE 
¡ESTA EL ESPACIO 

3VE QUE PUNTEROS PREVIOS 
3 DEBE ESTABLECER 


3 СОСЕ LA DIRECCION DE 
¡ENTRADA PREVIA AL OBJETO 
3Y LA COLOCA EN EL AREA 
3DEL PUNTERO 

3 RECOBRA LA DIRECCION DEL 
3 OBJETO 

¿LA РОМЕ EN LA POSICION 
3DEL PUNTERO 


¿LIMPIA EL STACK 

СОСЕ LA DIRECCION DEL 
3 INDICE 

3 CARGA HL EN EL 


3MUESTRA QUE HA HECHO 


3COGE LA DIRECCION DEL 
¿OBJETO 
3VE SI ESTA ALLI 


¿FIJA HL AL AREA DE PUNTERO 
5 DEL OBJETO 


3RECOBRA EL PUNTERO 


; BORRA EL MARCADOR DE ОСОР. 
3VE SI ES NECESARIO CAMBIAR 
3EL INDICE 


3PONE LA DIRECCION EN HL 
¿FIJA HL AL PUNTERO DE LA 


¡ENTRADA PREVIA 


Н INC HL 


0105 O1FFFF 1470 LD ВС, #OFFFF 


01D8 C9 1480 SALIRE RET 
1490 ; 
1500 ; 
1510 ; 
01D9 ES 1520 PRETAB PUSH HL 
01DA FD7EOO 1530 LD А, (1Ү+0) ; СОбЕ LA PRIMERA LETRA DEL 
1540 ; OBJETO 
O1DD 3D 1550 DEC A ¿BORRA LA CABECERA ASCII 
O1DE 0640 1560 SUB #40 
01ЕО CB27 1570 SLA A ¿MULTIPLICA POR 2 
01Е2 240401 1580 LD HL, (REFBAS) 
01Е5 85 1590 ADD A,L 
01Е6 ФЕ 1600 LD L,A 
O1E7 D2EBO1 1610 JP NC,FIJAR 
O1EA 24 1620 INC H 
O1EB EB 1630 FIJAR EX  DE,HL 
O1EC E1 1640 POP HL 
O1ED C9 1650 RET 
1660 ; 
O1EE 1670 FIN END 
BASETA 0102 BORRAR O1AC NOVALE 0146 NUEVO 015Е 
BUSQ 0108 CAMBIA O1CF РКЕТАВ 0109 REFBAS 0104 
COMPAR 011A ENCONT 015В SALIR O1AB SALIRE 01D8 
Figura 9.29 FIJAR O1EB FIN O1EE SETINX 0140 SIGUIE 0169 
Programas de la lista encadena- INDEXA 0100 LONENT 0106 TERMIN 0148 
da (continuación). MOVIN 0105 NOENCO 015D 


Relación de objetos 


Objetos en memoria y sus posiciones en 


memoria 
IM300 
0300 353 00 00 00 SO0N1111111111... 
ы 0310 44 32 00 00 00 Пап22222222229... 
9320 ап оо 00 00 М0М33533332232... 
9330 55 00 00 00  UN(:4444444444,,, 
0340 41 5 00 00 00 ANTS555555555... 
0350 41 6 00 00 00 446666666666... 
0360 41 5 00 00 00 1727777777777... 
0370 5з в оо 00 00 SID8888888888.,.. 
Carácter EOT en la 
tabla inicial 
“ПМ400 


0400 ?E 00 00 00 00 00 00 00-00 00 00 00 00 OO 00 OO ооо 
0410 00 00 00 00 00 00 00 00-00 00 00 00 OO 00 00 OO 0000000000 
0420 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ...........».... 
0430 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 eee mmn 
0440 00 00 00 00 00 00 00 00-00 00 00 00 OO 00 OO OQ ................ 
0450 00 00 00 00 оо оо оо 00-00 00 00 00 00 00 00 OO ...........».... 
0460 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 OO ................ 
0420 00 00 00 00 00 00 00 00-00 00 00 00 00 00 OO OO ................ 


-IM500 Directorio inicial 
0500 00 04 00 04 00 04 00 04-00 04 оо 04 00 04 оо O04 ................ 
0510 00 04 00 04 оо 04 00 04-00 04 OO 04 00 04 оо OA ................ 
0520 00 04 00 04 оо 04 00 04-00 04 00 04 00 04 00 04 ................ 
0530 00 ол оо 04 00 00 00 00-00 оо 00 00 оо оо 00 00 .............. у 

i 0540 00 00 00 00 00 00 00 00-00 00 00 OO 00 00 00 00 

paura id del d 0550 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 Қ 
ase de prueba del programa ae 0560 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ................ 


la lista encadenada. 0570 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ................ 


A e нинин ннн ыны нынын ын _ сте. -- 


Figura 9.30 

Pase de prueba del programa de 
la lista encadenada (continua- 
ción). 
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Marcadores de ocupación 


Punteros Configuración de la 
tabla tras varias 
inserciones 

ПМ400 


0400 00 00 00 <............... 


0410 5 35 25 35 35 25-35 35. 45 3 35 04 01 амт55 .. 
0420 2 32 32 32-32 32 › 04 01 Dane? 2222... 
0430 04 O1  ^^^6666566656565... 
0440 04 01 . 
2450 04 01 . 
0460 04 01 . 
0470 04 01 D 
бү 
Y:03650 310 
0226/2279 Eliminación de una entrada 


Ex02209 02297 


El único cambio 


09400 
0400 
9410 
0420 44 41 
0430 41 41 


00 
35 


32 32 
36 34 


0440 53 ағ 31 k К К 01 5041111111111... 
0450 ап ағ 33 22 33 33 33 33 00 04 01 МОМ32232333332... 
0450 53 49 28 38 за 38 38 40 04 01 511088888888888.. 
0470 41 са“ 32 37 37 37 37 00 04 01 1727777777777... 


-6220/223 


Pase de 'BUSQ” la entrada eliminada 


224 0224“ 


ün — № hallada 
< 
N A=37 EC=00FF ПЕ=0400 Н1.=0000 $=0100 Р=0223 02237 CALL 0171 
A'500 #7 #0000 D'70000 H’=0000 X=0400 Y=0310 1-00 (01717) 


-8Y 
Y-0310 340 


Pase de 'BUSQ” una entrada existente 


62207 


FOND 0228’ 


Hallada 


710 ПЕ<0430 HL=043E 9 


пк 
Z N A=54 RC 


0100 F 


223 02229 CALL 0171 


‘+00 R':0000 [”:0000 Н 0000 Xz0410 340 1:00 (0171) 
-6225/229 + . 
Eliminar Dirección de la entrada 
220229 02297 en la tabla 


Nota: Cambios en los 


-DM400 punteros 

0400 00 00 00 <............... 
0410 5 70 04 00 : ТӘН 
0420 2 00 04 00 2222222222... 
0430 70 04 01 AAA6666666666P.. 
0440 00 04 01 501111111111... 
0450 00 04 O1 МОМ3323323334... 
0460 40 04 01 SIDOBGBBSBOBBO.. 
0470 00 04 01 4222772777777... 


El programador principiante no tiene necesidad de preocu- 
parse todavía por los detalles de la realización y manipulación 
de estructuras de datos, aunque la programación eficiente de 
algoritmos no triviales obliga a dominarlas. Los ejemplos pre- 
sentados en este capítulo ayudarán al lector a conseguir ese 
dominio y a resolver todos los problemas planteados por las 
estructuras de datos habituales. 
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Desarrollo 


de programas 


Introducción 


Todos los programas que hemos visto hasta ahora los 
hemos desarrollado “a mano”, sin ayuda del soporte lógico ni 
del físico. La única mejora sobre el uso directo del código 
binario ha sido la escritura en un código mnemotécnico llama- 
do lenguaje ensamblador. Pero para desarrollar buenos progra- 
mas es necesario conocer las posibilidades que ofrecen los so- 
portes físico y lógico, y valorarlas es justamente la finalidad de 
este capitulo. 


Opciones básicas de programación 


Un programa puede escribirse fundamentalmente de tres 
formas: en código binario o hexadecimal, en lenguaje ensambla- 
dor o en un lenguaje de alto nivel. Analicemos una por una las 
tres posibilidades. 


CODIFICACION HEXADECIMAL 


Lo normal es escribir los programas en ensamblador, pero 
la mayor parte de los sistemas baratos de placa única no 
disponen del programa ensamblador encargado de traducir au- 
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tomáticamente los términos mnemotécnicos a código máquina, 
lo que obliga a efectuar esa traducción a mano. La codificación 
binaria es muy molesta de usar y está expuesta a errores, por lo 
que normalmente se prefiere la hexadecimal. Como ya vimos en 
el capitulo 1, una cifra hexadecimal representa cuatro bits bina- 
rios, de manera que el contenido de cualquier byte se representa 
con sólo dos. En el apéndice se encuentran los equivalentes 
hexadecimales de las instrucciones del 780. 

En resumen, si los recursos del usuario son limitados y no 
dispone de ensamblador, debe traducir a mano el programa a 
hexadecimal. Es una labor razonable si las instrucciones son 
pocas —entre 10 y 100, por ejemplo—, pero aburrida y propen- 
sa a errores si el programa es más largo; sin embargo, el hecho 
es que la mayor parte de los microordenadores de placa única 
obligan a trabajar en hexadecimal porque, para que salgan más 
baratos, se fabrican sin programa ensamblador y sin teclado 
alfanumérico. 

La codificación hexadecimal no es, pues, la más aconsejable, 
sino la más barata. E] precio de un ensamblador y de un 
teclado alfanumérico se compensa ampliamente con el trabajo 
que se ahorra al introducir el programa en memoria. En cual- 
quier caso, ello no altera la forma en que se escribe el programa 
propiamente dicho: esto se hace siempre en lenguaje ensambla- 
dor, para que sea comprensible por el programador humano. 


PROGRAMACION EN LENGUAJE ENSAMBLADOR 


Esta forma de programación se refiere tanto a los progra- 
mas que se cargan en código hexadecimal como a los que se 
introducen en código simbólico mnemotécnico. Vamos a cen- 
trarnos en este segundo caso, que exige disponer del correspon- 
diente programa ensamblador. Este lee cada una de las instruc- 
ciones simbólicas y las traduce al código binario utilizando de 1 
a 5 bytes, segün lo especificado al codificar las instrucciones. 
Pero un buen ensamblador dispone de otros recursos adiciona- 
les que facilitan la escritura de programas y que examinaremos 
más adelante. En particular, dispone de seudoinstrucciones que 
modifican el valor de los símbolos. Puede trabajarse con direc- 
cionamiento simbólico y saltarse a una posición simbólica. Du- 
rante la fase de puesta a punto, que suele suponer la elimina- 
ción o la adición de instrucciones, no es preciso reescribir el 
programa completo si se inserta alguna entre una bifurcación y 
el punto al que se bifurca, porque se emplean etiquetas simbóli- 
cas que el ensamblador ajustará automáticamente durante la 
traducción. Gracias al ensamblador, el programa puede también 


Figura 10.1 
Niveles de programación. 


ponerse a punto en forma simbólica. Para examinar el conteni- 
do de una posición de memoria y reconstruir la instrucción que 
representa a nivel ensamblador puede emplearse un desmonta- 
dor o desensamblador. Más adelante veremos los recursos que 
ofrece el soporte lógico de un sistema, pero antes vamos a 
detenernos en la tercera opción. 


POTENCIA 
DEL 
LENGUAJE 


APL 
COBOL 

FORTRAN ALTO NIVEL 
PL/M 

PASCAL 

BASIC 

MINI-BASIC 


MACRO 
CONDICIONAL NIVEL ENSAMBLADOR 


ENSAMBLADOR 


SIMBOLICO 


HEXADECIMAL/ 
OCTAL 
LENGUAJE MAQUINA 


BINARIO 


LENGUAJES DE ALTO NIVEL 


Los programas pueden, por último, escribirse en BASIC, APL, 
PASCAL u otro cualquiera de los numerosos lenguajes de alto 
nivel. Las técnicas de trabajo con éstos están fuera del alcance 
de este libro, y nos limitaremos aquí a reseñar brevemente 
algunas peculiaridades de las mismas. Los lenguajes de alto 
nivel constan de instrucciones potentes que hacen la labor de 
programación mucho más rápida y mucho más fácil. Dentro del 
ordenador hay un programa muy complejo que traduce dichas 
instrucciones a la representación binaria de máquina. Por lo 
general, a cada instrucción de alto nivel corresponden muchas 
binarias. El programa encargado de la traducción se llama 
compilador o intérprete. El compilador traduce primero el progra- 
ma entero a código objeto y a continuación lo ejecuta; el 
intérprete, por el contrario, traduce una instrucción y la ejecuta 
antes de pasar a la siguiente; tiene la ventaja de la interactivi- 
dad, pero a cambio de una eficacia inferior a la del compilador. 
No diremos nada más aquí de esta clase de lenguajes, y nos 
limitaremos a partir de ahora a la programación de un micro- 
procesador real en lenguaje ensamblador. 
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Veremos aquí los recursos lógicos más importantes que hay, 
o debe haber, en los sistemas de desarrollo completos. Ya 
hemos definido algunos de ellos; antes de seguir los describire- 
mos brevemente y definiremos los demás programas de interés, 

El ensamblador es el programa encargado de traducir la 
representación mnemotécnica de las instrucciones a su equiva- 
lente binario. А cada instrucción simbólica suele corresponder 
una binaria (que puede ocupar 1, 2 ó 3 bytes). El código binario 
resultante se llama código objeto y es directamente ejecutable 
por el microordenador. El ensamblador genera también un lis- 
tado simbólico completo del programa, las tablas de equivalen- 
cia que debe usar el programador y la lista de frecuencia de 
aparición de símbolos en el programa. Veremos algunos ejem- 
plos en este mismo capítulo. 

También hace el ensamblador una relación de errores de 
sintaxis, como instrucciones mal escritas o ilegales, fallos de 
bifurcación, etiquetas duplicadas o inexistentes, etc. 

Pero no elimina los errores lógicos, que son competencia 
exclusiva del programador. a 

El programa compilador traduce las instrucciones escritas en 
lenguaje de alto nivel a su forma binaria. 

El intérprete es parecido al compilador, ya que también 
traduce instrucciones de alto nivel a representación binaria, 
pero no conserva la representación intermedia de las mismas y, 
además, las ejecuta inmediatamente. Incluso es normal: que ni 
siquiera genere código intermedio y ejecute directamente las 
instrucciones de alto nivel. 

El monitor es un programa básico indispensable para utilizar 
los recursos físicos del sistema. Controla constantemente las 
entradas de los periféricos y organiza el resto de los dispositi- 
vos. Por ejemplo, en un microordenador de placa única equipa- 
do con teclado e indicadores LED, el monitor mínimo, debe 
vigilar constantemente el teclado por si se produce una entrada 
y presentar el contenido especificado en la pantalla LED; tam- 
bién debe comprender unas pocas órdenes introducidas por 
teclado, como ARRANCAR, PARAR, SEGUIR, CARGAR EN 
MEMORIA o EXAMINAR MEMORIA. En sistemas grandes 
suele llamarse al monitor programa ejecutivo, ya que también se 
encarga de efectuar manipulaciones complejas en los ficheros y 
de organizar las tareas. Todo este conjunto de recursos consti- 
tuye el sistema operativo; cuando los ficheros residen en disco, el 
sistema operativo se denomina sistema operativo de disco о 
DOS (disk operative system). 


La finalidad del editor es facilitar la introducción y modifica- 
ción de textos y programas. Permite al usuario introducir carac- 
teres cómodamente, suplementarlos, insertarlos, añadir líneas, 
eliminar líneas y buscar caracteres o series de caracteres. Es un 
recurso importante que facilita la introducción de textos. 

Cuando un programa no funciona bien, lo normal es que no 
haya ninguna indicación del origen del fallo, y por eso el 
programador suele insertar puntos de interrupción que lo detie- 
nen en direcciones especificadas para poder examinar el conteni- 
do de la memoria en esos puntos; ésta es, justamente, la función 
principal del programa de puesta a punto (debugger). Permite 
interrumpir un programa, volver a ponerlo en marcha, exami- 
narlo y visualizar y modificar el contenido de los registros de 
memoria. Los buenos programas de puesta a punto también 
analizan datos en forma simbólica, hexadecimal, binaria u otra 
representación habitual, y los cargan en esos mismos formatos. 

El programa de carga es responsable de la colocación de 
varios bloques de código objeto en posiciones especificadas de 
memoria y del ajuste de sus respectivos apuntadores simbólicos 
para que puedan referenciarse mutuamente. Se emplea para 
desplazar programas o bloques a diversas áreas de memoria. El 
programa simulador o emulador imita el funcionamiento de un 
dispositivo, por lo general el microprocesador, en su ausencia 
cuando se desarrolla un programa sobre un procesador simula- 
do antes de cargarlo en la placa real. De esta forma se puede 
suspender un programa, modificarlo y archivarlo en memoria 
ВАМ; pero el simulador también tiene inconvenientes: 


1. Por lo general, sólo simula el procesador, no los dispositi- 
vos de entrada/salida. 

2. La velocidad de ejecución es baja y opera en tiempo 
simulado, lo que impide la verificación de dispositivos de 
tiempo real y puede plantear problemas de sincronización 
aun cuando la lógica del programa sea correcta. 


Un emulador es, básicamente, un simulador de tiempo real. 
Utiliza un procesador para simular otro con todo detalle. 

Se llaman rutinas de servicio a las necesarias en la mayor 
parte de las aplicaciones y de las que el usuario quisiera ver 
encargarse al fabricante: multiplicación, división y otras opera- 
ciones aritméticas, desplazamiento de bloques, verificación de 
caracteres, manipulación de dispositivos de entrada/salida, etc. 
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Vamos a analizar la secuencia típica de desarrollo de un 
programa a nivel de ensamblador. Supondremos que dispone- 
mos de todos los recursos mencionados antes, con el fin de 
poner de relieve su utilidad. Cuando no se encuentran en un 
sistema, siempre pueden reemplazarse por los correspondientes 
programas, pero a costa de un mayor esfuerzo de puesta a 
punto. 

Lo normal es empezar por crear un algoritmo y definir las 
estructuras de datos del problema que desea resolverse. А conti- 
nuación hay que desarrollar todos los diagramas de flujo nece- 
sarios; y, por último, convertir éstos en un programa escrito en 
lenguaje ensamblador (esta última fase se llama codificación). 

Acto seguido se introduce el programa en el ordenador, 
para lo que existen diversos recursos físicos que estudiaremos 
en la siguiente sección. 

El programa pasa a la memoria RAM del sistema bajo la 
supervisión del editor. Cuando ha entrado una parte completa 
del mismo —una o varias subrutinas—, se comprueba su fun- 
cionamiento. 

Primero se utiliza el programa ensamblador. Si no reside en 
el sistema, se carga a partir de una memoria externa, como un 
disco. Á continuación el programa se ensambla, es decir, se 
traduce a código binario, lo que da lugar al programa objeto, 
listo ya para ser ejecutado. 

Es raro que un programa funcione bien a la primera. Para 
verificar su buen comportamiento se introducen en posiciones 
cruciales, en las que sea fácil comprobar si los resultados inter- 
medios son correctos, una serie de puntos de interrupción. Para 
efectuar la comprobación se emplea el programa de puesta a 
punto. Al dar la orden “ARRANQUE”, el programa se pone en 
marcha y se detiene en los puntos especificados, que el progra- 
mador aprovecha para comprobar el contenido de los registros, 
o de la memoria, y asegurarse de que los datos son correctos; 
en caso afirmativo, sigue hasta el punto siguiente. Si aparece un 
dato incorrecto, es que hay un error en el programa; lo primero 
que hace en este caso el programador es repasar el listado para 
ver si los códigos están bien escritos; si lo están, cabe suponer 
que el error es de naturaleza lógica, y en ese caso conviene 
estudiar los diagramas de flujo (estamos suponiendo que éstos 
se han comprobado previamente a mano y funcionan razona- 
blemente bien). Lo normal es que el fallo esté en la codificación, 
lo que obliga a rehacer un segmento del programa. Si la repre- 
sentación simbólica del mismo está todavía en memoria, no hay 
más que reintroducir el editor y modificar las líneas afectadas, 


para a continuación repetir de nuevo la sección corregida. En 
algunos sistemas la memoria no es suficientemente capaz, y es 
preciso llevar la representación simbólica del programa a un 
disco o una cinta antes de ejecutar el código objeto; en este 
caso, como es natural, hay que volver a cargar la representación 
simbólica a partir del soporte antes de introducir de nuevo el 
editor. 

El procedimiento descrito se repite todas las veces que sea 
necesario hasta que el programa funcione correctamente. Es 
preciso insistir en que la prevención es mucho más eficaz que la 
curación. Si el diseño es correcto, el programa empezará a 
funcionar bien rápidamente en cuanto se corrijan los errores 
mecanográficos inevitables o los fallos de codificación obvios; 
por el contrario, poner a punto un programa mal diseñado 
lleva muchísimo tiempo, más todavía que el empleado en el 
diseño. Por tanto, vale más dedicar más tiempo a éste y reducir, 
en cambio, el de corrección. 

Pero, aunque de esta forma se pone a prueba la organiza- 
ción general del programa, no se verifica el funcionamiento en 
tiempo real con dispositivos de entrada/salida. Si hay que hacer 
esta verificación, lo más inmediato es cargar el programa en 
EPROM, instalarlo en la placa y observar si funciona. 

Pero hay una solución mejor: consiste en utilizar un circuito 
emulador interno que utiliza el microprocesador Z80 (o cual- 
quier otro) para emular al Z80 casi en tiempo real. La emula- 
ción es física; el dispositivo está equipado con una línea acaba- 
da en un conector de 40 patillas, dispuestas exactamente igual 
que las del Z80, que se acopla a la placa real de la aplicación en 
la que se está trabajando. Las señales generadas por el emula- 
dor son idénticas a las del Z80, aunque quizá un poco más 
lentas. La ventaja más importante es que el programa en estu- 
dio sigue residiendo en la RAM del sistema de desarrollo y 
genera las señales reales de comunicación con los dispositivos 
reales de entrada/salida que quieran utilizarse. De esta manera 
puede trabajarse en el programa con todos los recursos del 
mencionado sistema de desarrollo (edición, puesta a punto, 
recursos simbólicos, fichero) y a la vez comprobar la entrada/sa- 
lida en tiempo real. 

Pero un buen emulador no se limita a esto, sino que ofrece, 
además, otras posibilidades, como el analizador, que registra las 
últimas instrucciones o el estado de los diversos buses del siste- 
ma antes de un punto de interrupción; en otras palabras, el 
analizador hace un resumen de los acontecimientos ocurridos 
antes del punto de interrupción o del error; incluso puede 
examinar una dirección determinada o la aparición de una 
combinación de bits especifica. Es un recurso muy valioso, 
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Figura 10.2 
Un mapa de memoria típico. 
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porque cuando se encuentra un error suele ser demasiado tarde, 
puesto que la instrucción o el dato que lo han provocado son 
anteriores a la detección. El analizador permite al programador 
localizar el segmento causante del error; si el segmento analiza- 
do no fuese suficientemente largo, bastaría situar antes el punto 


de interrupción. 


Con esto termina la descripción de la serie de operaciones 
de que consta el desarrollo de un programa. Pasemos ahora a 
describir los recursos físicos que facilitan dicho desarrollo. 
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DEL USUARIO 


Recursos físicos 


MICROORDENADOR MONOPLACA 


El microordenador de una sola placa es la forma más bara- 
ta de iniciarse en el desarrollo de programas. Normalmente 
dispone de un teclado hexadecimal, unas pocas teclas de funcio- 
nes y 6 indicadores LED que visualizan direcciones y datos. 
Dado que la memoria es reducida, el aparato suele carecer de 
ensamblador; en el mejor de los casos tiene un pequeño moni- 
tor y apenas recursos de edición y puesta a punto, con excep- 
ción de unas pocas órdenes; por tanto, todos los programas 
deben introducirse en forma hexadecimal, y así es también 
como aparecen en los indicadores LED. En teoría, un microor- 
denador monoplaca tiene la misma potencia física que cualquier 
otro, y si no soporta los recursos habituales de un sistema 
mayor y alarga mucho el desarrollo de programas es únicamen- 
te por lo reducido de su memoria. Como trabajar en sistema 
hexadecimal es muy pesado, los microordenadores monoplaca 
son adecuados, sobre todo, para estudiar programas educativos 
y de adiestramiento, habitualmente bastante cortos. Constitu- 
yen, sin duda, la forma más económica de aprender a progra- 
mar practicando, pero para desarrollar con ellos aplicaciones 
complejas es imprescindible conectarles pastillas de memoria 
adicionales capaces de albergar los recursos lógicos habituales. 


SISTEMAS DE DESARROLLO 


Un sistema de desarrollo es un microordenador con una 
cantidad considerable de memoria RAM (32K o 48K) y los 
dispositivos de entrada/salida adecuados: pantalla, impresora, 
discos y, habitualmente, un programador de PROM, además de 
un circuito emulador. Es un aparato creado especificamente 
para facilitar el desarrollo de programas en el medio industrial. 
Normalmente ofrece todos o casi todos los recursos lógicos 
mencionados en la sección anterior, por lo que, en principio, 
constituye el instrumento de desarrollo de programas idóneo. 

Tiene el inconveniente de que, por lo general, no soporta 
compilador ni intérprete, que normalmente ocupan mucha me- 
moria, más de la que posee el sistema. De todas formas, ofrece 
todo lo necesario para crear programas en lenguaje ensambla- 
dor. Lo malo es que, como se vende mucho menos que los 
ordenadores para aficionados, cuesta bastante más caro. 
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MICROORDENADORES PARA AFICIONADOS 


El soporte físico de uno de estos microordenadores es exac- 
tamente igual al de un sistema de desarrollo. La diferencia 
principal radica en que normalmente no disponen de los refina- 
dos recursos lógicos instalados en los aparatos industriales. Asi, 
cuentan con ensambladores elementales, y editores y sistemas de 
ficheros mínimos, y carecen de programador de PROM, de 
circuito emulador interno y de sistemas potentes de puesta a 
punto. Representan, pues, una opción intermedia entre un mi- 
croordenador monoplaca y un sistema de desarrollo. Para 
quien desee crear programas de complejidad modesta constitu- 
yen, sin duda, el mejor compromiso entre la economía de 
adquisición y una cantidad aceptable de instrumentos de desa- 
rrollo. 


SISTEMAS EN TIEMPO COMPARTIDO 


Varias firmas alquilan terminales conectados a redes de 
tiempo compartido que utilizan grandes ordenadores y aprove- 
chan sus ventajas. Casi todos estos sistemas disponen de ensam- 
bladores cruzados para todos los microordenadores. Un ensambla- 
dor cruzado no es más que un ensamblador para el procesador X 
que reside en el procesador Y (por ejemplo: un ensamblador 
para el Z80 instalado en un IBM 370). El tipo del ordenador 
central carece de importancia. El usuario escribe sus programas 
en el ensamblador del Z80, y el ensamblador cruzado los traduce 
al código binario apropiado. La diferencia estriba en que el 
programa no puede ejecutarse más que en un procesador simu- 
lado y siempre que no utilice recursos de entrada/salida. Es, 
pues, una solución limitada al medio industrial. 


ORDENADOR GRANDE 


Si se tiene acceso a un ordenador grande y potente, también 
puede emplearse un ensamblador cruzado. Cuando el ordenador 
está disponible en tiempo compartido, la opción es básicamente 
igual a la anterior. Pero el servicio por lotes constituye uno de 
los medios de desarrollo de programas más incómodo. porque 
el tiempo de trabajo se alarga muchísimo. 


¿CON PANEL FRONTAL O SIN PANEL FRONTAL? 


El panel frontal es un accesorio del soporte físico que suele 
utilizarse en la puesta a punto de programas, tradicionalmente 
para visualizar cómodamente el contenido binario de un regis- 
tro o de la memoria. Sin embargo, todas sus funciones puede 
ejecutarlas un terminal, y la actual pantalla de rayos catódicos 
ofrece un servicio similar al del panel, con la ventaja de que en 
ella es fácil pasar de representación binaria a hexadecimal, 
simbólica o decimal (siempre que se disponga de las correspon- 
dientes rutinas de conversión, por supuesto). El inconveniente es 
que para obtener la visualización deseada hay que pulsar varias 
teclas en lugar de girar un botón. De todas formas, como el 
panel es bastante caro, la mayor parte de los microordenadores 
actuales han prescindido de este dispositivo de puesta a punto. 
Su valor se defiende, con frecuencia, más con argumentos senti- 
mentales, justificados por la experiencia pasada del programa- 
dor, que con razones reales. Desde luego, no es imprescindible. 


RESUMEN DE RECURSOS FISICOS 


A grandes rasgos, cabe distinguir tres situaciones: Quien 
dispone de un presupuesto minimo y desea aprender a progra- 
mar deberá adquirir un microordenador monoplaca que le per- 
mita desarrollar todos los programas sencillos de este libro y 
muchos más. Sus limitaciones se hacen sentir en cuanto los 
programas superan unos pocos cientos de instrucciones. 

El usuario industrial necesitará un sistema de desarrollo, ya 
que cualquier otro medio más pobre en recursos alargará consi- 
derablemente el tiempo de trabajo necesario para poner a pun- 
to los programas. La equivalencia está clara: a mejor soporte 
físico, menos tiempo de programación. Naturalmente, para desa- 
rrollar programas sencillos puede seguirse un procedimiento 
más barato, pero si los programas son complejos no tiene 
sentido escatimar en la adquisición de recursos físicos, porque 
la parte más costosa del desarrollo será justamente la progra- 
mación. 

Para el aficionado bastarán los recursos mínimos, pero sufi- 
cientes, que proporcionan los microordenadores domésticos. 
Todavia faltan por desarrollar muchos recursos lógicos de desa- 
rrollo para tales microordenadores. y el usuario deberá evaluar 
su sistema a la vista de las observaciones hechas en este capí- 
tulo. 

Pasemos ya a analizar con detalle el recurso más importante 
de todos: el programa ensamblador. 
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El programa ensamblador 
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A lo largo de todo el libro hemos estado utilizando el 
lenguaje ensamblador sin hablar de la síntesis formal del mismo 
ni definirlo. La finalidad de este lenguaje es proporcionar al 
usuario una representación simbólica fácil de usar y que, a la 
vez, pueda traducir rápidamente el correspondiente programa 
ensamblador a la notación binaria. 


CAMPOS DEL ENSAMBLADOR 


He aquí los campos utilizados al escribir un programa en 
lenguaje ensamblador: 


—el campo de etiqueta, opcional, que puede contener una 
dirección simbólica para la instrucción que sigue; 

— el campo de instrucción, que contiene el código de opera- 
ción y los operandos (puede establecerse un campo de 
operandos independiente); 

— el campo de comentarios, situado a la derecha, opcional y 
encaminado a facilitar la lectura del programa. 


Los tres aparecen en el formulario de programación de la 
figura 10.3. 

Cuando el ensamblador recibe el programa, confecciona un 
listado del mismo, operación que conlleva la creación de tres 
nuevos campos, situados, por lo general, a la izquierda de la 
página (véase el ejemplo de la figura 10.4). Cada una de las 
líneas escritas por el programador recibe un nümero simbólico 
que se escribe a la izquierda del todo. 

El siguiente campo avanzando hacia la derecha es el campo 
de la dirección real, que recoge en formato hexadecimal el valor 
del contador del programa que señala la instrucción de que se 
trate. 

A su derecha se encuentra la representación hexadecimal de 
la instrucción. 

Esto sugiere una de las posibles aplicaciones del programa 
ensamblador, aunque, cuando se trabaje con un microordena- 
dor monoplaca que sólo acepte el código hexadecimal, puede 
pasarse el programa escrito en ensamblador por un sistema que 
disponga del correspondiente programa, si se tiene acceso al 
mismo, que generará la codificación hexadecimal correcta. 


S3NOIOVAH3S80 


OYTOANIS 


оамуыза0 | олцунзао 


091409 
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Теје е 


TVWIO3GQVX3H 
NOI99NHLSNI 


Formulario de programación de 
un microprocesador. 


Figura 10.3 
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TABLAS 


Cuando traduce el programa simbólico a notación binaria, 
el ensamblador realiza dos tareas esenciales: 


1. Pasa las instrucciones mnemotécnicas a código binario. 
2. Pasa los símbolos que representan constantes y direccio- 
nes a notación binaria. 


Para facilitar la puesta a punto del programa, el ensambla- 
dor presenta al final del listado la equivalencia entre los símbo- 
los y su valor hexadecimal; es lo que se llama la tabla de 
simbolos. 

Algunas tablas no se limitan a recoger el símbolo y su valor, 
sino que, además, indican los números de línea en que aparecen 
dichos símbolos. 


MENSAJES DE ERROR 


Durante la traducción, el ensamblador detecta los errores de 
sintaxis y los incluye en el listado final. Son diagnósticos habi- 
tuales los siguientes: símbolos sin definir, etiquetas ya definidas, 
código de operación, direcciones o modos de direccionamiento 
incorrectos. Naturalmente, es deseable disponer de diagnósticos 
más detallados, como los que proporcionan algunos programas 
ensambladores. 


EL LENGUAJE ENSAMBLADOR 


Ya hemos definido los códigos de operación, así que descri- 
biremos aquí los símbolos, las constantes y los operadores que 
pueden utilizarse como parte de la sintaxis del ensamblador. 


SIMBOLOS 


Sirven para representar valores numéricos, sean datos o 
direcciones. Pueden estar formados por hasta seis caracteres, y 
deben comenzar por uno alfabético; los otros cinco sólo pueden 
ser letras del abecedario o nümeros; además, no pueden utili- 
zarse los nombres de los códigos de operación, los de los 
registros (А, B, C, D, E, H, L, BC, DE, HL, АЕ, ВС, DE, IX, IY 
y SP) ni los de los seudooperadores utilizados por el programa 
ensamblador (los nombres de estas seudooperaciones aparecen 
más adelante, en la sección correspondiente). Tampoco está 
permitida la utilización como símbolos de las denominaciones 
de las banderas: C, Z, N, PE, NC, P, PO, NZ y M. 


Figura 10.4 
Ejemplo de salida del ensambla- 
dor. 


Asignación de un valor a un símbolo 


Las etiquetas son símbolos especiales cuyo valor no tiene 
que definir el programador, ya que será automáticamente asig- 
nado por el programa ensamblador conforme las vaya encon- 
trando. De esta forma, el valor de la etiqueta corresponde 
automáticamente a la dirección de la instrucción generada en 
la línea en que se encuentra. Hay seudoinstrucciones especiales 
para forzar nuevos valores de partida a las etiquetas o para 
asignarles valores específicos. 


0100 10 ORG #0100 
0100 0002 20 MPRAD DEFW #0200 
0102 0202 30 MPDAD БЕРИ #0202 
0104 0402 40 RESAD DEFW #0204 
50 ; 
0106 Е0480001 60 MP488 LD BC, (MPRAD) ¿CARGA MULTIPLICADOR EN C 
010A 0608 70 LD B,8 3B ES CONTADOR DE BIT 
010C Ер5В0201 80 LD DE, (MPDAD) ¿CARGA MULTIPLICANDO EN E 
0110 1600 90 LD D,O 3 INICIALIZA D 
0112 210000 100 LD HL, O РОМЕ A O EL RESULTADO 
0115 CB39 110 MULT SRL C ¿SHIFT AL ACARREO DEL BIT 
115 ; 3 MULTIPLICADOR 
0117 3001 120 JR NC, NOADD 3 COMPRUEBA EL ACAREO 
0119 19 130 ADD HL,DE ¿SUMA AL RESULTADO MPD 
011A CB23 140 NOADD SLA Е ¿SHIFT IZQUIERDA DE MPD 
011С CB12 150 RL D ¿GUARDA EL ВІТ EN D 
011Е 05 160 DEC B 3DECR EL CONT DE SHIFT 
O11F С21501 170 JP NZ,MULT JREPETIR SI CONTADOR< >O 
0122 220401 180 LD (RESAD) , HL 3ALMACENA EL RESULTADO 
190 END 


Por el contrario, el programador debe definir, antes de usarlos, 
los valores asignados a los símbolos correspondientes a cons- 
tantes y direcciones de memoria. 

La asignación de valor se realiza por medio de seudoinstruc- 
ciones, que son instrucciones para el ensamblador que no se 
traducen en otras ejecutables. Así, la constante LOG se defini- 
ría: 


LOG EQU 3002H 


la sentencia asigna el valor hexadecimal 3002 a la variable 
LOG. En una próxima sección estudiaremos las seudoinstruc- 
ciones del ensamblador. 


CONSTANTES O LITERALES 
Tradicionalmente, las constantes pueden expresarse en nota- 


ción decimal, hexadecimal, octal o binaria o como series de 
caracteres alfanuméricos. La base de representación empleada se 
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indica mediante un símbolo. Para cargar “0” en el acumulador 
basta escribir 


LD А,0 


aunque opcionalmente puede añadirse una “D” al final de la 
constante. 

Los números hexadecimales terminan con el símbolo “Н”. 
Para cargar el valor “FF” en el acumulador, se escribe: 


12  A,0FFH 


Los valores octales terminan con los símbolos “O” o “Q”, y 
los binarios, con el “В”. 

Por ejemplo, para cargar en el acumulador el valor 
“11111111”, se escribe: 


LD A,11111111B 


En el campo literal son también admisibles los caracteres 
literales ASCII, que deben ir encerrados entre comillas simples. 
Así, para cargar el símbolo “S” en el acumulador se escribe: 


LD А, 5 


Ejercicio 10.1: ¿Cargarán el mismo valor en el acumulador las dos 
instrucciones LD A, 5 y LD A, 5H? 


Obsérvese que en la convención de Zilog los paréntesis 
denotan direcciones. Por ejemplo: 


LD А, (10) 


especifica que el acumulador se carga con el contenido de la 
dirección decimal de memoria (10). 


OPERADORES 


Los operadores son recursos del ensamblador que facilitan 
la escritura de programas simbólicos. Hacen falta como mínimo 
los signos más y menos para poder especificar, por ejemplo: 


LD  A,(DIRECCION) 
LD  A,(DIRECCION + 1) 


Es importante darse cuenta de que la expresión DIREC- 
CION + 1 será utilizada por el ensamblador para determinar la 
dirección real de memoria que debe utilizar como equivalente 
binario; es decir, la expresión se calcula durante la fase de 
ensamblado, no durante la ejecución del programa. 

Puede haber otros operadores, como la multiplicación y la 
división, útiles para acceder a tablas de memoria. O algunos 
más especializados, como mayor que y menor que, que truncan 
valores de dos bytes en sus componentes inferior y superior. 

Naturalmente, las expresiones deben dar lugar a valores 
positivos. Normalmente no se emplean números negativos, que, 
en todo caso, deben representarse en formato hexadecimal. 

Por último, se emplea el símbolo especial “$” para represen- 
tar el valor en curso de la dirección de la línea; debe interpre- 
tarse como “posición actual” (valor del PC). 


Ejercicio 10.2: ¿Qué diferencia hay entre las dos instrucciones 
siguientes? : 


LD А,10101010В 
LD А, (101010108) 


Ejercicio 10.3: ¿Cuál será el resultado de la siguiente instrucción? : 


JR NC,$-— 2 


EXPRESIONES 


El ensamblador del Z80 permite utilizar gran diversidad de 
expresiones con operaciones aritméticas y lógicas. Para calcular- 
las, avanza de izquierda a derecha, con arreglo a las prioridades 
que recoge la tabla de la figura 10.5. Para forzar un orden de 
evaluación determinado, puede utilizarse paréntesis, pero siem- 
pre teniendo en cuenta que los más externos indicarán que lo 
contenido entre ellos deberá tratarse como una dirección. 


SEUDOINSTRUCCIONES DEL ENSAMBLADOR 


Se llama así a las órdenes especiales que da el programa- 
dor al ensamblador y que determinan el almacenamiento de 
valores en símbolos, o en memoria, o el control de la ejecución 
del programa, o de su impresión. Las que controlan precisa- 
mente la impresión se llaman también “órdenes”, y las describi- 
remos en una sección aparte. 
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Figura 10.5 
Tabla de prioridad de los ope- 
radores. 
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Veamos ahora las 11 seudoinstrucciones con que cuenta el 
sistema de desarrollo Zilog: 


ORG nn 


iguala el contador de direcciones del ensamblador al valor mn; 
en otras palabras; la primera instrucción ejecutable que aparez- 
ca tras esta seudoinstrucción residirá en el valor nn; sirve para 
alojar diferentes segmentos del programa en posiciones de me- 
moria específicas. 


EQU nn 
asigna un valor a una etiqueta. 
DEFL nn 


también asigna un valor nn a una etiqueta, pero puede repetirse 
dentro del programa con diferentes valores para la misma, al 
contrario que EQU, que sólo puede usarse una vez. 


DEFB n 


asigna un contenido de 8 bits al byte que reside en el contador 
de referencia en curso. 


OPERADOR FUNCION PRIORIDAD 


+ MAS UNARIO 

MENOS UNARIO 

NO LOGICO 

RESULTADO 

POTENCIACION 
MULTIPLICACION 

DIVISION 

MODULO 

DESPLAZAMIENTO LOGICO DCHA. 
DESPLAZAMIENTO А LA 120. 
SUMA 

RESTA 

Y LOGICO 

O LOGICO 

O EXCLUSIVO LOGICO 
IGUAL 


„МОТ. or Y 


.АМО. or & 
.OR. ог 1 
.XOR. 

.EQ. or = 
„СТ. or > MAYOR QUE 

AT. ог < MENOR QUE 

.UGT. MAYOR QUE SIN SIGNO 
.ULT. MENOR QUE SIN SIGNO 


1 
1 
1 
1 
2 
3 
3 
3 
3 
3 
4 
4 
5 
6 
6 
7 
7 
7 
7 
7 


DEFB ‘5’ 
asigna el valor ASCII de "S" al byte 


DEFW nn 


asigna el valor nn a la palabra de dos bytes que reside en el 
contador de referencia en curso y en la posición siguiente. 


DEFS nn 


reserva un bloque de memoria de nn bytes de tamaño a partir 
del valor en curso del contador de referencia. 


DEFM ‘5’ 


almacena en memoria la serie “S”, que empieza en el contador 
de referencia en curso; debe tener una longitud inferior a 63. 


MACRO PO РІ... Ри 


sirve para definir una etiqueta como macro, asi como para 
determinar su lista formal de parámetros; describiremos las 
macros más adelante. 


END 


indica el final del programa; el ensamblador ignorará todas las 
sentencias escritas a continuación, 


ENDM 


señala el fin de la definición de una macro. 


ORDENES DEL ENSAMBLADOR 


Las órdenes se usan para modificar el formato del listado y 
controlar la impresión del mismo. Todos emplezan con un 
asterisco en la primera columna. El ensamblador del 280 dispo- 
ne de siete, de las que las más típicas son: 


*EJECT 
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que hace que el listado pase a la parte superior de la página 
siguiente; y 


*LIST OFF 


que determina la interrupción de la impresión. Las otras cin- 
со son: “*HEADING S", “*LIST ON", “*MACLIST ON”, 
“*MACLIST OFF", **INCLUDE FILENAME". 


MACROS 


Una macro —o macroinstrucción— no es sino un grupo de 
varias instrucciones. Resulta muy cómodo para el programador 
poder sustituir una serie de instrucciones, que se utiliza muchas 
veces de la misma forma, por una macro que las represente, ya 
que se ahorra el trabajo de escribirlas todas las veces. Así, el 


grupo 


SAVREG MACRO 
PUSH AF 
PUSH BC 
PUSH DE 
PUSH HL 
ENDM 


puede sustituirse a partir de este momento por “SAVREG”. 
Cada vez que en el programa aparece SAVREG se ejecutan las 
cinco instrucciones a las que representa. Los ensambladores que 
tienen esta posibilidad se llaman macroensambladores, y se 
limitan a hacer una mera sustitución física de líneas equivalen- 
tes cada vez que encuentran el nombre de la macro. 


¿Macro o subrutina? 


А primera vista parece que una macro funciona como una 
subrutina, pero no es así. Cuando se utiliza el programa ensam- 
blador para obtener el código objeto en el listado, la macroins- 
trucción se reemplaza por las instrucciones reales que la compo- 
nen. Durante la ejecución, el grupo completo se repetirá tantas 
veces como su nombre genérico. 

Por el contrario, la subrutina se define una sola vez, y el 
programa salta a su dirección cada vez que se utiliza. La macro 
es un recurso de tiempo de ensamblado, mientras que la subruti- 
na lo es de tiempo de ejecución. Su comportamiento es muy 
diferente. 
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Parámetros macro 


Cada macro puede equiparse con una serie de parámetros. 
Veamos un ejemplo: 


SWAP MACRO НМ, НМ, ЯТ 

LD А, HM М EN A 

LD HT, А А EN T (= M) 
LD A, HN NENA 

LD #M, À А EN M (= М) 
LD A, HT TENA 

LD HN. А А EN N (=T) 
END M 


Esta macroinstrucción intercambia los contenidos de las po- 
siciones de memoria M y N, una operación con la que no 
cuenta el 780, y que puede realizarse con una macro. En este 
caso, “T” es simplemente el nombre de una posición de almace- 
namiento temporal que necesita el programa. Vamos, por ejem- 
plo, a intercambiar los contenidos de las posiciones de memoria 
ALFA y BETA: 


SWAP (ALFA), (BETA), (TEMP) 


En esta instrucción, TEMP es el nombre de una posición 
temporal de almacenamiento que sabemos que está disponible y 
que puede usar la macro. La descomposición de ésta en sus 
componentes elementales sería: 


LD  A,(ALFA) 
LD (TEMP) А 
LD  A,(BETA) 
LD (ALFA) А 
LD  A,(TEMP) 
LD (ВЕТА),А 


Como es fácil comprobar, para el programador resulta útil 
"utilizar seudoinstrucciones definidas con macros, porque ello le 
permite ampliar en apariencia las instrucciones del 780. No 
obstante, hay que tener en cuenta que la ejecución se hace 
instrucción por instrucción, de modo que las macros funcionan 
más lentamente que las instrucciones aisladas. De todos modos, 
son muy útiles para desarrollar programas largos. 
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Otros recursos macro 


A la simple posibilidad de crear macroinstrucciones se le 
pueden añadir recursos sintácticos y seudoinstrucciones adicio- 
nales, que permiten, por ejemplo, hacer macros incluidas, es 
decir, contenidas en una llamada macro más general. Si dentro 
de ésta se incluye una definición que modifique a la propia 
macro, ésta producirá en una primera llamada una expansión, y 
otra diferente en llamadas posteriores. El ensamblador del 780 
dispone de esta posibilidad, pero no de la de incluir definiciones 
dentro de otras. 


Ensamblador condicional 
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Es otro de los recursos que proporciona el Z80. Gracias a 
él, el programador puede diseñar programas para casos muy 
diversos y ensamblar condicionalmente sus segmentos según el 
caso. Por ejemplo: un usuario crea un programa para controlar 
los semáforos de un cruce con diversos algoritmos de mando. Á 
continuación recibe las instrucciones del ingeniero de tráfico lo- 
cal, en las que se especifican los semáforos y los algorítmos que 
deben emplearse. El programador se limita a fijar los paráme- 
tros en el programa general y ensamblarlo condicionalmente, 
operación que dará lugar a un programa “a la medida”, que 
conservará únicamente las rutinas necesarias para resolver el 
problema específico que se le plantea. 

Este recurso es útil en medios industriales, que cuentan 
siempre con varias opciones y que hacen interesante para el 
programador la posibilidad de montar rápida y automática- 
mente segmentos de programas en respuesta a los parámetros 
externos. 

La versión del microensamblador del Z80 que proporciona 
Zilog sólo cuenta con dos seudooperaciones condicionales: 


COND NN y ENDC 


siendo NN una expresión. La seudooperación “COND NN” 
determina la evaluación de NN: si el resultado es positivo (no 
0), se incluye la instrucción que sigue a COND; pero, si es nulo, 
se eliminan todas las instrucciones hasta la aparición de ENDC. 

ENDC señala el final de COND y permite el ensamblado de 
todas las instrucciones que siguen. Estas seudooperaciones no 
pueden incluirse dentro de sí mismas. 
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Resumen 


En teoría podría haber más posibilidades condicionales con 
las especificaciones “IF” y “ELSE”; quizá estén presentes en 
futuras versiones del ensamblador. 


Hemos visto en este capítulo las técnicas y los recursos 
físicos y lógicos que forman parte del desarrollo de los progra- 
mas junto con diversas equivalencias y alternativas. 

Estas van desde el microordenador monoplaca hasta un 
auténtico sistema de desarrollo, por lo que respecta al soporte 
físico, y desde el código binario hasta los lenguajes de progra- 
mación de alto nivel, por lo que respecta al lógico. El lector 
deberá hacer la elección que considere óptima en función de sus 
recursos e intereses. 
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Conclusión 


Hemos recorrido todos los aspectos importantes de la pro- 
gramación, desde las definiciones y conceptos básicos hasta la 
manipulación interna de los registros del Z80, pasando por el 
control de los dispositivos de entrada/salida y los recursos 
lógicos de desarrollo. ¿Cuál es el siguiente paso? Hay dos pers- 
pectivas, una relacionada con el desarrollo de la tecnología, y la 
otra con el desarrollo de los conocimientos y la experiencia del 
propio programador. Examinémoslas brevemente, 


Desarrollo tecnológico 


El avance de la integración con tecnología MOS permite 
fabricar pastillas cada vez más complicadas. Paralelamente, el 
precio de producir el procesador propiamente dicho no deja de 
reducirse, de manera que en la actualidad casi todas las pastillas 
de entrada/salida y de control de periféricos montan procesado- 
res sencillos y son programables, lo que plantea un dilema 
interesante: para simplificar el desarrollo. de programas y para 
reducir el número de componentes, las nuevas pastillas de E/S 
disponen de refinados recursos programables que integran mu- 
chos algoritmos; en consecuencia, el desarrollo de programas 
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viene a complicarse, en contra de lo esperado, porque el progra- 
mador debe estudiar en detalle todas estas nuevas pastillas. 
Programar un sistema ya no es programar el microprocesador 
únicamente, sino también todas las pastillas conectadas a él, y el 
tiempo necesario para llegar a dominar cada una de ellas puede 
ser considerable, 

Pero el dilema es sólo aparente, porque, si no existiesen tales 
pastillas, la complejidad de las conexiones por realizar y de los 
correspondientes programas sería todavía mayor. La nueva 
complicación radica en la necesidad de programar más de un 
solo procesador y de estudiar las peculiaridades de cada una de 
las pastillas que componen el sistema. No obstante, cabe espe- 
rar que las técnicas y conceptos expuestos en este volumen 
hagan la tarea razonablemente fácil. 


El siguiente paso 
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Ya se han estudiado las técnicas necesarias para programar 
aplicaciones sencillas, que era el objetivo de este libro. El paso 
siguiente es practicar, algo que no puede reemplazarse por 
ningún texto. Es imposible aprender a programar sólo estudian- 
do; hay que practicar y adquirir experiencia. Ahora está en 
situación de escribir sus propios programas, y espero que esa 
nueva andadura le sea propicia. 


re 


APENDICE A 


TABLA DE CONVERSION HEXADECIMAL 


л, 9 10 11 12 13 14 15 
17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 
33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 
49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 
65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 
81 82 83 84 85 86 B7 88 89 90 91 92 93 94 95 
97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 


113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 
129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 
145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 
161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 
177 178 179 180 181 132 183 184 185 186 187 188 189 190 191 
193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 
209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 
225 226 227 228 229 230 231 232 233 234 235 236.237 238 239 
241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 


nMmOOD }» © © Jo» MAIN + о 


PAN rs 2 


со 


Е 14,680,064| Е 917,504| Е 57,344| Е 3,584 | Е 224 | Е 
F 15,728,640] Е 983,040] Е 61,440] Е 3,840 | Е Е 


БЕГЕН 
0 010 оро 0 0 
1 1,048,576] 1 65,536] 1 4,09 1 256 
2 2,097,152] 2 131,072] 2 8,1921 2 512] 2 32 
3 3,145,728] 3 196,608] 3 12,288] 3 768| 3 48 
4 
5 


2 
3 
5 5,242,880] 5 327.680] 5 в. 5 
7 7,340,032| 7 458,752| 7 28,672 1,792 | 7 m2 
8 8,388,608] 8 524,288] 8 32,768] 8 2,044| 8 128 | 8 8 
9 9,437,184| 9 us ei 9 2304 9 144|9 9 
А 10,485,760] А 655360| А 40,90] А 2560 А 160 A 10 
D 13,631,488| D 851,968) D 53,248] D 3,328] D — 208| D 13 
} , 14 


APENDICE B 


TABLA DE CONVERSION ASCII 


2lw 
gje 
o[o 
alo 
alu 


2 
000 001 010 011 1 101 110 111 


0 
1 
2 
3 
4 
5 
6 
7 
8 
9 
A 
B 
с 
D 
E 
F 


-- © ©  O M ADN © 
мх сс о-оо роо 


+ 
3 


OZZTAC-=-IONTMOOD>O 
1 — 


1>---м<х=<ечозоэ 
О23-<х----га чоаооср 1 


әу! A 
o 
m 


SIMBOLOS ASCII 


NUL — Nulo 
SOH — Principio de encabezamiento 
STX — Principio de texto 
: ETX — Final de texto 
ENQ — Pregunta 
ACK — Reconocimiento 
BEL — Timbre 


BS —Retroceso 

HT ——Tabulación horizontal 

LF  — Alimentación de línea 

VT — Tabulación vertical 

FF  — Alimentación de formato 

CR  — Retorno de carro 

SO  — Desplazamiento hacia afuera 
SI — Desplazamiento hacia adentro 


DLE — Cambio de enlace de transmisión 
DC — Control de dispositivo 

NAK — Reconocimiento negativo 

SYN — Funcionamiento síncrono 

ETB — Fin de transmisión de bloque 
CAN — Cancelar 

EM — Fin de medio 

SUB — Sustituir 

ESC — Cambiar 


. 


FS  — Separador de ficheros 
GS  — Separador de grupos 
RS  — Separador de registros 
US — Separador de unidades 
SP  — Espacio (blanco) 

DEL — Borrar 
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APENDICE C 


TABLAS DE BIFURCACION RELATIVA 


TABLA DE BIFURCACION RELATIVA HACIA ADELANTE 


D E 


13 14 
27 28 29 30 
36 37 38 39 40 41 42 43 44 45 


52 53 54 55 56 57 58 59 60 61 
68 69 70 71 72 73 74 75 76 77 
84 85 86 87 88 89 90 91 92 93 94 
99 100 101 102 103 104 105 106 107 108 109 10 
13 114 15 16 17 18 119 120 121 122 123 124 125 126 


TABLA DE BIFURCACION RELATIVA HACIA ATRAS 


1 2 3 4 5 6 7 8 9 A B с D E 


127 126 125 124 123 122 121 120 119 118 117 16 115 114 
111 110 109 108 107 106 105 104 103 102 10: 100 99 98 
95 94 93 92 91 90 89 88 87 86 85 
79 78 77 76 74 73 72 71 70 69 
63 62 60 59 58 57 56 55 54 53 52 51 50 
46 45 44 43 42 4 40 39 38 37 36 35 34 

31 30 29 28 27 26 25 24 23 22 21 20 19 18 
15 14 13 12 n 10 9 8 7 6 5 4 3 2 


Slo 
© 
N 
м“ 
ВЕ 


12 
96 
80 
64 
48 
32 
16 


127 


© = 
зо [т 


83 82 81 
67 66 65 


zje a 
міо ©] 


APENDICE D 


CONVERSION DECIMAL A BCD 


DECIMAL BCD DEC BCD 
p ----- 
0 0000 10 00010000 
1 0001 11 00010001 
2 0010 12 00010010 
3 0011 13 00010011 
4 0100 14 00010100 
5 0101 15 00010101 
6 0110 16 00010110 
7 0111 17 00010111 
8 1000 18 00011000 
9 1001 19 00011001 


DEC BCD 
90 10010000 
91 10010001 
92 10010010 
93 10010011 
94 10010100 
95 10010101 
96 10010110 
97 10010111 
98 10011000 
99 10011001 
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APENDICE E 


CODIGOS DE LAS INSTRUCCIONES DEL 780 
(la letra d equivale en el código objeto a 05) 


CODIGO INSTRUCCION CODIGO INSTRUCCION 
OBJETO FUENTE OBJETO FUENTE 


8E A,(HL) E620 
008Е05 A,(1X+d) CB46 O,(HL) 
FD8E05 A,(1Y *d) DDCB0546 0,(1X+d) 
8F АА FDCB0546 0.(1Y+d) 
88 А,В CB47 ОА 
89 АС CB40 ов 
8А АО CB41 ос 
8B A,E CB42 0,D 
8C АН CB43 ОЕ 
80 AL CB44 0,H 
CE20 Ал СВ45 од 
ED4A HL,BC CB4E 1 (HL) 
ЕОБА HL,DE DDCB054E 1,(1X+d) 
ED6A HL,HL ЕОСВО54Е 1,(1Y+d) 
ED7A HL,SP СВ4Е ТА 
86 АДНО CB48 1.8 
008605 A Xd) CB49 с 
FD8605 A1 Y *d) CB4A 1D 
87 A,A CB4B 1,E 
80 A,B свас 1H 
81 АС CB4D 1L 
82 AD CB56 2.(HL) 
83 AE DDCB0556 2,(1X+d) 
84 АН FDCBO556 2,1Y*d) 
85 AL CB57 2,А 

Ал CB50 2.8 
09 HL,BC CB51 2.с 
19 HL,DE CB52 2,D 
29 HL,HL CB53 2,E 
39 HL,SP CB54 2,H 
0009 IX,BC CB55 2,L 
DD19 IX,DE CB5E 3,(HL) 
DD29 хх ООСВО55Е 3,1X*d) 
DD39 IX,SP FDCBOSSE ^ S,0Ysd) 


ЕО09 1Y,8C СВЕ З,А 
FD19 ¡Y DE CB58 3,8 
FD29 ҮЛҮ СВ59 3,C 
FD39 ¡Y SP CB5A 3,0 

A6 (HL) CB5B 3,E 
004605 (1+9) CB5C 3,H 
FDA605 (Yd) CB5D 3,L 

A7 CB66 4,(HL) 
AO DDCB0566 4,1X*d) 
A1 FDCBO566 4,0Y*d) 
A2 CB67 4,A 

A3 CB60 ав 

А4 CB61 4C 

A5 CB62 40 
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CODIGO 
OBJETO 


CB63 
CB64 
CB65 
CB6E 
DDCBO56E 
FDCBO56E 
CB6F 
C868 
CB69 
C86A 
CB6B 
CB6C 
CB6D 
CB76 
DDCB0576 
ЕОСВО576 
CB77 
CB70 
CB71 
CB72 
CB73 
CB74 
CB75 
CB7E 
ООСВО57Е 
ЕОСВОБ7Е 
CB7F 
CB78 
CB79 
CB7A 
CB7B 
CB7C 
CB7D 
DC8405 
FC8405 
D48405 
C48405 
F48405 
EC8405 
E48405 
CC8405 
CD8405 
3F 

BE 
DDBEO5 
FDBEO5 
BF 

B8 

B9 

BA 

B8 

BC 

BD 


FE20 
EDA9 
EDB9 


INSTRUCCION 
FUENTE 


BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
BIT 
CALL 
CALL 
CALL 
CALL 
CALL 
CALL 
CALL 
CALL 
CALL 
ССЕ 
СР 
СР 
СР 
СР 
СР 
СР 
СР 
СР 
СР 
СР 
СР 


4E 

4,H 

4L 
5,(HL) 
5,(1X+d 
5,(1Y+d) 
5,A 

5,8 

5,C 

5,0 

5,E 

5,H 

GL 

6. (HL) 
6,(1X+d) 
6,(1Y+d) 
6,A 

6,B 

6,C 

6,D 

6,E 

6,H 

6,L 
7,(HL) 
7.(1X+d) 
7,0Y ға) 
7,A 

7.8 

7,C 

7.0 

7,Е 

7,H 

74 
C,nn 
M,nn 
NC,nn 
М2 nn 
P,nn 
PE,nn 
PO,nn 
Zinn 

nn 


(HL) 
(1X*d) 
(1Y+d) 


CODIGO 
OBJETO 


EDB1 
EDA1 
2Е 

27 

35 
203505 
[03505 
3D 

05 

ов 

00 

15 

18 

10 

25 

2B 
DD2B 
FD2B 


ED46 
ED56 
ED5E 
ED78 
ED40 
ED48 
ED50 
ED58 
ED60 
ED68 
34 
DD3405 
FD3405 
3c 
04 
03 
oc 
14 
13 
1C 
24 
23 


INSTRUCCION 
FUENTE 


CPIR 
CPI 
CPL 
DAA 
DEC 
DEC 
DEC 
DEC 
DEC 
DEC 
DEC 
DEC 
DEC 
DEC 
DEC 
DEC 
DEC 
DEC 
DEC 
DEC 
DI 
DJNZ 
EI 
EX 
EX 
EX 
EX 
EX 
EXX 
HALT 
IM 
IM 
IM 
IN 
IN 
IN 
IN 
IN 
IN 
IN 


(HL) 
(1X+d) 
(1Y+d) 
A 


(SP), HL 
(SP),IX 
(SP), ЛУ 
АҒ,АҒ” 
DE,HL 


0 

1 

2 

А. (С) 

вас) 

САС) 

D,(C) 

ЕС) 

H,(C) 

С.С) 

(HL) 

(1X*d) 
(1Y+d) 
A 


CODIGO INSTRUCCION CODIGO INSTRUCCION 
OBJETO FUENTE OBJETO FUENTE 


EDAA DD7E05 A,(1X+d) 
EDBA FD7EO5 A, Y *d) 
EDA2 3A8405 A.(nn) 
EDB2 7F LD A,A 
C38405 nn 78 LD AB 

E9 (но 79 Lo АС 
DDE9 axi 7A LD АО 
FDE9 uy) 7B LD АЕ 
DA8405 Cnn 7C LD А.Н 
FAB405 M,nn ED57 LD Al 
D28405 NC,nn 7D LD AL 
C28405 NZ nn 3E20 LD An 
F28405 P.nn EDSF LD AR 
ЕА8405 РЕ пп ar. LD вн) 
Е28405 PO,nn 204605 LD B,(1X+d) 
СА8405 Z,nn FD4605 LD B.(1Y+d) 
382€ Се 47 LD B,A 
302€ NC.e 40 LD в.в 
202Е NZ.e a1 LD B.C 


282€ Ze 42 LD 8,0 
182E e МЕ 43 LD B,E 


02 (8C),A 44 LD B.H 
12 (DEA 45 LD BL 


77 IHLA 0620 LD B,n 

70 (нов ED4B8405 LO BC (пп) 
71 (нс 018405 го BC,nn 
72 ТЕЙ») 4Е LD C.(HL) 
73 (НОЕ DD4E05 LD C.(1X+d) 
74 (нон FD4E05 LD C.(IY+d) 
75 (HL).L 4F LD СА 
3620 (нНОл 48 LO св 
DD7705 (1X*d),A 49 го сс 
DD7005 (1X+d),8 4A LD CD 
DD7105 (1X*d),C 48 LD CE 
DD7205 (1X+d),D 4C LD сн 
007305 (1X*d),E 40 LD CL 
DD7405 (1X*d),H 0E20 LD Сл 
007505 (1X*d),L 56 LD D,(HL) 
00360520 (1X*d),n DD5605 LD Diu Xd) 
FD7705 (1 Y *d),A FD5605 LD D, Y *d) 
FD7005 (Y *d),B | 57 LD D,A 
FD7105 (1 Y*d),C 50 LD D.B 
FD7205 UY+d),D 51 LD D.C 
FD7305 (IY+d),E 52 LD D,D 
FD7405 (1Y+d),H 53 LD D,E 
FD7505 (1Y+d),L 54 LD D.H 
FD360520 (1+9) п 55 LD D,L 
328405 (nnl, A 1620 LD D,n 
ED438405 {nn),BC ED5B8405 LD ОЕ (пп) 
ED538405 (nn),DE 118405 LD ОЕ пп 
228405 (nn), HL 5E LD E,(HLI 
DD228405 (пп) 1X DD5E05 LD E.(1X+d) 
FD228405 (nn), Y ЕОБЕОБ LD E,(1Y*d) 
ED738405 (nn), SP 5F LD ЕА 

ОА АВС) 58 LD E.B 

1A A,IDE) 59 LD ЕС 

7Е АДНО 5A LD ЕО 


591 


592 


CODIGO 
OBJETO 


5B 

5С 

5D 
1E20 
66 
DD6605 
FD6605 
67 


60 
61 


62 


63 
64 


65 

2620 
2A8405 
218405 
ED47 
DD2A8405 
00218405 
FD2A8405 
FD218405 
6Е 
DD6E05 
FD6E05 


6F 
68 


69 

6A 

6B 

6c 

6D 
2E20 
ЕО4Е 
Е0788405 
F9 
DDF9 
FDF9 
318405 
ЕОАВ 
EDB8 
EDAO 
EDBO 
ED44 
00 

B6 
008605 
FDB605 
B7 

BO 

B1 

B2 

B3 

B4 

B5 


INSTRUCCION 
FUENTE 


E,E 

Е,Н 

EL 

En 
нн) 
H,(1X+d) 
H,(IY+d) 
НА 


H.B 
H,C 


но 


НЕ 
нн 


HL 
На 

НЕ, (nn) 
HL, nn 
LA 

ІХ (пт) 
IX,nn 
IY (пп) 
1У пп 
L,(HL) 
L,(IX+d) 
L,(1Y+d) 


LA 
(в 


L,C 
го 
L,E 
н 
Lt 
гл 
В,А 
SP (nn) 
SP,HL 
SP,IX 
SPY 
SP. nn 


(HL) 
(Хей) 
(1Y*d) 
A 


CODIGO 
OBJETO 


EDB3 
ED79 
ED41 
ED49 
ED51 
ED59 
ED61 
ED69 
D320 
EDAB 
EDA3 

F1 

C1 

D1 

E1 

DDE1 
FDE1 

F5 

C5 

D5 

ES 

DDES 
ЕОЕБ 
CB86 
DDCB0586 
FDCB0586 
CB87 
CB80 
c881 
CB82 
CB83 
CB84 
CB85 
CB8E 
DDCBO58E 
FDCB058E 
CB8F 
CB88 
CB89 
CB8A 
CB8B 
CB8C 
CB8D 
C896 
DDCB0596 
FDCB0596 
CB97 
C890 
C891 
C892 
C893 
C894 
CB95 
CB9E 
DDCBO059E 
FDCBO59E 


INSTRUCCION 
FUENTE 


OTIR 
OUT 
OUT 
OUT 
OUT 
OUT 
OUT 
OUT 
OUT 
OUTD 
OUTI 
POP 
POP 
POP 
POP 
POP 
POP 
PUSH 
PUSH 
PUSH 
PUSH 
PUSH 
PUSH 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 


(C),A 
(C),B 
(CI,C 
(C),D 
(C),E 
(C),H 
(Сл 
(n), A 


AF 

BC 

DE 

HL 

ІХ 

ІҮ 

АҒ 

BC 

DE 

HL 

IX 

IY 
O,(HL) 
0,(1X+d) 
0,(1Y+d) 
О,А 
0,В 

ос 

0,0 

0,Е 

он 

OL 
1,(HL) 
1.0X«d) 
LAY ed) 
1,A 

1,8 

1,C 

1,0 

1,Е 

н 

14 
2,(HL) 
2,1X*d) 
2,(1Y+d) 
2,А 

2.8 

2,C 
2,0 
2,Е 

2.H 

2;L 
3,(HL) 
3,(1X*dl 
3,(1Y+d) 


CODIGO 
OBJETO 


CB9F 
CB98 
CB99 
CB9A 
CB9B 
CB9C 
C89D 
CBA6 
DDCB05A6 
РОСВОБАб 
CBA7 
СВАО 
CBA1 
CBA2 
СВАЗ 
СВА4 
СВАБ 
СВАЕ 
DDCBOSAE 
РОСВОБАЕ 
CBAF 
СВАВ 
СВА9 
СВАА 
СВАВ 
СВАС 
СВАО 
CBB6 
DDCB05B6 
FDCB05B6 
C887 
CBBO 
CBB1 
CBB2 
CBB3 
CBB4 
CBB5 
CBBE 
ООСВОБВЕ 
РОСВОБВЕ 
СВВЕ 
CBB8 
CBB9 
CBBA 
CBBB 
CBBC 
CBBD 

C9 

D8 

F8 

00 


INSTRUCCION 
FUENTE 


RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RES 
RET 
RET 
RET 
RET 
RET 
RET 
RET 
RET 
RET 


3,A 

3,B 

3,C 

3,D 

3,E 

3,H 

3,L 
4,(HL) 
4,(1X*d) 
4,(1Y+d) 
4,A 

4,8 

4,0 

4,0 

4,E 

4,H 

4 
5,(HL) 
5,(1X+d) 
5,(1Y+d) 
5,A 

5,В 

5,С 

5,D 

5,E 

5,H 
БЕ 
6,(HL) 
6,(1X*d) 
6,(1Y+d) 
6,A 
6.8 

6,C 
6,D 

6,E 

6.H 
6,L 
7,(HL) 
7,(1X*d) 
7,(1Y*d) 
7,A 

7,8 
лс 

7,D 

7E 

7,H 

74 


CODIGO INSTRUCCION 
OBJETO FUENTE 


ED4D 
ED45 

CB16 (HL) 
DDCB0516 (хе 
FDCBO516 (уча) 
CB17 

CB10 

c811 

CB12 

CB13 

CB14 

CB15 

17 

своб 

00680506 

FDCB0506 

CB07 

cB00 

CB01 

сво2 

своз 

CB04 

сво 

07 

ED6F 

СВТЕ 

DDCB051€ 

ЕОСВОБТЕ 

СВ1Е 

CB18 

CB19 

СВТА 

свтв 

CB1C 

CB1D 

1F 

СВОЕ (HL) 
DDCBOSOE (1X*d) 
ЕОСВОБОЕ (1Y+d) 
СВОЕ 

свов 

сво9 

СВОА 

свов 

свос 

CBOD 

ОЕ 

ED67 

с? 

СЕ 

р? 

DF 

E7 

EF 

F7 

FF 

DE20 


593 


594 


CODIGO 
OBJETO 


9E 
DD9E05 
FD9E05 
9F 

98 

99 

9A 

9B 

9c 

9D 

ED42 
ED52 
ED62 
ED72 

37 

CBC6 
DDCBO5C6 
ЕОСВ05С6 
CBC7 
CBCO 
CBC1 
CBC2 
CBC3 
CBC4 
CBC5 
CBCE 
DDCBOSCE 
FDCBOSCE 
CBCF 
CBC8 
CBC9 
CBCA 
CBCB 
CBCC 
CBCD 
своб 
DDCB05D6 
ЕОСВ0506 
CBD7 
CBDO 
CBD1 
CBD2 
CBD3 
CBD4 
CBD5 
CBD8 
CBDE 
DDCBO5DE 
ЕОСВОБОЕ 
СВОЕ 
сво9 
СВОА 
CBDB 
CBDC 
CBDD 
CBE6 


INSTRUCCION 
FUENTE 


A,IHL) 
A,(1X+d) 
А ГУ+а) 
A,A 

A,B 

A,C 
A,D 
AE 

AH 

AL 
HL,BC 
HL,DE 
HL,HL 
HL,SP 


O,(HL) 
0,(1X+d) 
0,(1Y +d} 
0,А 

0,8 

ос 

0,D 

0,Е 
он 
0. 
1,(HL) 
1,(1X+d) 
1,(1Y+d) 
ТА 

1,8 

1c 

1,0 

1,Е 

1,H 

ТА 
2,(HL) 
2,(0X*d) 
2,(Y*d) 
2,A 

2.8 

2,C 
2.0 

2,Е 

2.н 
2% 

3,8 
3,(HL) 
3,(1X+d) 
3,(1Y+d) 
3,A 

3,C 

3,D 

3,E 

зн 

3,L 
4,(HL) 


CODIGO 
OBJETO 


DDCBO5E6 
FDCBOSEG 
CBE7 
CBEO 
CBE1 
CBE2 
CBE3 
CBE4 
СВЕБ 
СВЕЕ 
DDCBO5EE 
РОСВОБЕЕ 
СВЕЕ 
THES 
CBE9 
CBEA 
CBEB 
CBEC 
CBED 
CBF6 
DDCBO5F6 
FDCBO5F6 
CBF7 
CBFO 
CBF1 
CBF2 
СВЕЗ 
СВЕ4 
CBF5 
CBFE 
DDCBOSFE 
FDCBOSFE 
CBFF 
CBF8 
CBF9 
CBFA 
CBFB 
CBFC 
CBFD 
CB26 
DDCB0526 
FDCB0526 
CB27 
CB20 
C821 
CB22 
CB23 
CB24 
CB25 
СВ2Е 
DDCBO52E 
ЕОСВОБ2Е 
СВ2Е 
CB28 
CB29 
CB2A 


INSTRUCCION 
FUENTE 


4,(1X+d) 
4,(1Y+d) 
4,A 
4,8 

4,с 
4,D 
4,Е 

4,H 

4,L 
5,(HL) 
5,(1X*d| 
5,(1Y+d) 
5,A 

5,8 

5,C 

5,0 

5,Е 

5,H 

5,L 
6,(HL) 
6,0 X*d) 
6,(1Y+d) 
6,A 

6,8 

6,C 
6.0 

6,E 

6,H 

6,L 

7, (HL) 
7,(1X+d) 
7,0 Y*d) 
7,A 

7,8 

7,C 

7,0 

7,Е 

7,H 

7% 
(HL) 
(1X+d) 
(1Y+d) 


(HL) 
(1X+d) 
(IY+d) 
A 


B 
с 
D 


CODIGO INSTRUCCION 
OBJETO FUENTE 


C828 
свгс 
CB2D 
CB3E 
DDCBO53E 
ЕОСВОБЗЕ 
СВЗЕ 
CB38 
CB39 
CB3A 
CB38 
CB3C 
CB3D 

96 
DD9605 
FD9605 
97 

90 

91 

92 

93 

94 

95 

D620 

AE 
DDAEO5 
FDAEOS 
AF 

A8 

A9 

AA 

AB 

AC 

AD 
EE20 


(Cortesía de Zilog Inc.) 
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APENDICE F 


EQUIVALENCIAS DEL Z80 AL 8080 


180 


АРСА, (HL) ADCM EX (SP), HL XTHL ORn ORI [B2] 
АССА, n ACI [B2] HALT HLT ORr ОВА г 
АССА, г АОС г INA, (п) IN [B2] OR (HL) ORAM 
ADDA, (HL) ADDM INC BC INXB OUT (п), A OUT [B2] 
ADDA, n AD! [B2] INC DE INX D POP AF РОР PSW 
ADDA, r ADDr INC HL INX H POP BC POPB 
ADDHL BC DADB ІМС т INR г РОР DE РОРО 
ADDHL DE DADD INC SP INX SP POP HL POPH 
ADDHL HL — DADH INC (HL) INRM PUSH AF PUSH PSW 
ADDHL SP X DADSP JPC, nn JC [B2] [B3] PUSH BC PUSH В 
AND n ANI [B2 JP M, nn JM [B2][B3] PUSH DE PUSHD 
ANDr ANA г JP NC, nn JNC [B2] [B3] PUSH HL PUSH Н 
AND (HL) ANAM JP nn JMP [B2] [B3] RET RET 
CALLC, nn CC [B2] [B3] JP NZ, nn JNZ [B2] [B3] RETC RC 

CALL M, nn CM [B2] [B3] JPP, nn JP [B2] [B3] RETM RM 

САЦ МС, nn СМС (82) [B3] JP PE, nn JPE [B2][B3] RET NC 

CALL nn CALL JP PO, nn JPO [B2][B3] RET NZ 

САЦ NZ, nn СМ2 [B2] (83) JPZ, nn JZ [82] (83) RETP RP 


CALLP, nn CP (82) (B3] JP (HL) PCHL RET PE 


RET PO RPO 
[B3] RETZ RZ 
RLA RAL 
RLCA RLC 
RRA RAR 
RRCA RRC 
RST P RSTP 
SBC A, (HL) SBB M 
SBCA, n SBI [B2] 
SBC А, r SBB r 
SCF STC 
SUB n SUI [B2] 
SUB r SUB г 
SUB (HL) SUB M 
XOR п ХЕ! [B2] 
XORr XRAr 
XOR (HL) XRAM 


САЦ РЕ, nn CPE [B2] [83] LD A, (DE) LDAX 
САЦ PO, пп СРО [B2] [B3] LDA, (nn) LDA [B2 
САЦ 7, nn CZ [B2] [B3] LD DE, nn LXID, [B2] [83] 
ССЕ CMC LD SP, nn LXI SP, [B2] [B3] 
CPr CMPr LD (BC), A STAX B 

CP (HL) CMPM LD (DE), A STAX D 

CPL CMA LD (HL), r MOVM, г 

CPn CPI (B2] LD (nn), A STA [B2] [83] 
DAA DAA LD (nn), HL SHLD [B2] [B3] 
DEC BC DCXB ША, (BC) LDAX B 

DEC DE DCXD 1D BC, nn LXIB, [B2] [B3] 
DEC HL DCXH LD HL, (nn) LHLO (B2] [B3] 
DEC r DCRr LD HL, nn LXI H [B2] [B3] 
DEC SP DCX SP LD r, (HC) MOV 1, М 

DEC (HL) DCRM Шуп ММ! r, [B2] 

DI LDr, т MOV г], r2 

El LD SP, HL SPHL 

EX DE, HL МОР МОР 
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APENDICE G 


EQUIVALENCIAS DEL 8080 AL 780 


ACI [B2] ADCA, n IN [B2] INA, (n) POP H POP HL 
ADCM ADCA, (HL) RM INC (HL) POP PSW POP AF 
ADC г АССА, г INR г INCr PUSH В PUSH BC 
ADDM ADD A, (HL) INXB INC BC PUSHD PUSH DE 
ADDr ADDA, r INXD INC DE PUSH Н PUSH HL 
AD! [B2] ADDA, n INX H INC HL PUSH PSW PUSH AF 
ANAM AND (HL) INX SP INC SP RAL RLA 
ANA г ANDr JC [B2] [83] ЈРС, nn RAR RRA 
ANI [B2] AND n JM [B2] [83] JP M, nn RC RETC 
CALL CALL nn JMP (82) (83) JP nn RET RET 
CC (82) [B3] ^ САЦ С, nn JNC [82] [83] JP NC, nn RLC RLCA 
СМ [B2] [B3] САЦ M, nn JNZ [82] (83) ЈР М2, nn RM RETM 
CMA CPL JP [B2] [B3] ЈРР, nn RNC RET NC 
CMC CCF JPE 182) [B3] ЈРРЕ, nn RNZ RET NZ 
CMP M CP (HL) JPO [B2] [B3] JP PO, nn RP RETP 
CMP r CPr JZ [62] [83] ЈР2, nn RPE RET PE 
CNC [B2] [83] САЦ МС, nn LDA [82] [83] LDA, (пп) RPO RET PO 
СМ2 [B2] [83] САЦ М2, nn LDAX B LD À, (BC) RRC RRCA 
СР [В2] [83] ^ CALLP, nn LDAX D LDA, (DE) RST RSTP 
СРЕ [82] [83] САЦ PE, nn LH LD [82] [83] LD HL, (nn) RZ RETZ 
CPI [B2] CPn LXI B [82] [83] LD BC, nn SBBM SBC A, (HL) 
СРО [B2] (83) САЦ PO, nn 1000 [B2] [B3] LD DE, nn SBBr SBCA, г 
CZ [B2] [B3] ^ САИ Z, nn LXI H [B2] [B3] LD HL, nn 581 [B2] SBC А, n 
DAA DAA LXI SP [B2] [B3] LD SP, nn SHLD [B2] [B3] LD (nn), HL 
DADB ADD HL, BC MOVM, r LD (HL), r SPHL LD SP, HL 
DADD ADD HL, DE МОМ г, M 10 г, (HL) STA (82) [B3] LD (пп), А 
DADH ADD HL, HL MOV rl, r2 Dr, r' STAX В LD (BC), A 
DAD SP ADD HL, SP MVI M LO (HL), n STAXD LD (DE), A 
DCRM DEC (HL) ММ r [B2] Шел STC SCF 
DCRr DECr NOP NOP SUBM SUB (HL) 
DCX B DEC BC ORAM OR (HL) SUB. SUB г 
DCXD DEC DE ОВА г ORr SUI [B2] SUBn 
DCXH DEC HL ORI [B2] ORn XCHG EX DE, HL 
DCX SP DEC SP OUT [B2] OUT (п), A XRAM XOR (HL) 
DI DI PCHL JP (HL) ХВА г XORr 
El El POPB POP BC ха! [B2] XOR n 
POP D POP DE XTHL EX (SP), HL 


Indice 


alfabético 


Abreviado, direccionamiento, 408, 
414, 449. 

Absoluto, direccionamiento, 103, 
407, 413. 

Acarreo, 21, 23, 25-28, 30, 170-171. 

Acceso indirecto a la memoria, 470. 

ACT, 58. 

Acumulador, 406. 

Acumulador de 16 bits, 100. 

ADC, 95. 

ADC, A, s, 185. 

ADC HL, ss, 187. 

ADD, 92. 

ADD A, (HL), 76, 189. 

ADD A, (IX + d), 190. 

ADD А, (IY + d), 192. 

ADD A, n, 65, 194. 

ADD A, r, 64, 71, 72, 195. 

ADD HL, ss, 196. 

ADD IX, rr, 197. 

ADD IY, rr, 199. 

Alfanumérico, dato, 35-37. 

Algoritmo, 13-14, 109, 515, 

ALU, 44, 72, 78. 

Ambigüedad sintáctica, 14. 

Analizador, 561-562. 

AND, 163-164. 


AND s, 201. 

Ampliado, direccionamiento, 156, 
407-408, 413. 

Aplicaciones, 493, 

Arboles, 520-521. 

Aritméticos, programas, 90-101. 

Arquitectura básica, 44-46. 

Arquitectura estándar, 46. 

Arquitectura del sistema, 44. 

ASCII, 35-37, 499-500. 

ASCII, tabla de conversión, 36. 

Asignación de valor, 569, 

Asincrono, 441, 468, 490. 

Auxiliar, registro, 56, 58. 


B, 60. 
Bancos de registros, 60. 
Bandera de interrupciones, 182. 
Banderas, 28, 47-48, 170-175. 
Barrido de estaciones, 436, 439, 
464, 494, 520. 
BASIC, 22. 
BCD, 32-34, 499. 
aritmética, 101. 
banderas, 107. 
representación, 32. 
resta, 104. 
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зита, 101, 104. 

tabla, 32. 

transferencia de bloques, 506. 
BCD condensado, 32, 101. 
Biblioteca de subrutinas, 145. 
Biestable, 48. 
Bifurcación, instrucción de, 408. 


Binario, 18-22, 37, 40. 


Binario directo, 17. 

Binario con signo, 21-22. 

Bit, 16, 18, 37. 

BIT b, (HL), 203. 

ВІТ b, (IX + д), 205. 

ВТ b, (IY + d), 207. 

BIT b, r, 209. 

Bit de filtración de interrupciones, 
470. 

Bit de paridad, 36. 

Bits de estado, 48, 484. 

Bloque, 516, 518, 520. 

Bloques de acceso, 519. 

Borrado de memoria, 493. 

Bucle de barrido, 463-464. 

Bucle del programa, 61, 115. 

Bucle de retraso, 432, 454. 

Burbujeo, 509-513. 

Bus de control, 44. 

Bus de datos, 484. 

Bus de direcciones, 44. 

Búsqueda, 525, 533, 546. 

Búsqueda binaria, 522, 533-537, 
546-547. 

Búsqueda logarítmica, 522, 536. 

Búsqueda secuencial, 522. 

BUSRQ, 86, 468. 

Byte, 16-17, 37, 411. 

Byte superior, 98. 


C, 25, 27-30, 60, 70. 

Cálculo del tiempo, 433. 
CALL, 140, 153, 412, 471. 
CALL cc, pq, 211. 

CALL pq, 213. 

CALL SUB, 138-140. 

Campo de comentarios, 566. 
Campo de desplazamiento, 409. 
Campos del ensamblador, 566. 
Carácter de borrado, 437. 
Carácter de interrupción, 437. 
Carga, 91, 100. 

Casilla de control, 46. 

CCF, 215. 

Cero, 173. 

Ciclo de ejecución, 52. 

Ciclo de máquina, 66. 

Ciclo de memoria, 53. 


Ciclos de reloj, 66. 

Cifra binaria, 16. 

Clases de instrucciones, 151. 

Codificación, 14. 

Codificación hexadecimal, 38-39, 
555-556. 

Código binario, 17. 

Código incorrecto, 102. 

Código de operación, 63, 80, 406, 
410, 412. 

Cola, 519. 

Coma flotante, representación en, 
34-35, 

Comparar, 507. 

Compilador, 521, 557. 

Complemento a dos, 23-25, 27. 

Complemento a uno, 23. 

Comprobación de intervalos, 497- 
498. 

Conceptos básicos, 13. 

Conclusión, 579. 

COND, 577. 

Constantes, 407, 411, 570. 

Contacto, 448-449, 484. 

Contador, 433-434, 

Contador de datos, 49. 

Contador del programa, 49-50. 

Control, 436. 

Control E/S, 86-87. 

Controladores de los segmentos, 
452. 

Conversión de código, 499-500. 

CP, 163. 

CP s, 216. 

СРР, 218. 

CPDR, 219. 

CPI, 221. 

CPIR, 222. 

CPL, 129, 161, 224. 

CPU, 44, 182 

Cristal de cuarzo, 45. 

Cronómetro, 435. 

Cuenta de ceros, 504, 

Cuenta de impulsos, 435. 


D, 60, 69. 

DAA, 102, 225, 

Dato listo, 439. 

DEC m, 227. 

DEC rr, 229. 

DEC IX, 230. 

DEC IY, 231. 
Decimal, 18-19. 
Decodificación, 53, 68, 80. 
Decremento, 161, 409. 
DEFB, 573. 


DEFL, 572. 

DEFM, 573. 

DEFS, 573. 

DEFW, 573. 

Desarrollo de programas, 555, 560. 

Desarrollo tecnológico, 579. 

Desbordamiento, 27-30. 

Desbordamiento negativo, 30. 

Desplazamiento, 47, 60, 113, 114, 
152, 153, 

Desplazamiento aritmético, 114, 

Desplazamiento lógico, 114. 

Detección de impulsos, 435. 

Devolver, 457. 

DI, 232. 

Diagrama de flujo, 15-16, 108-109, 
417, 433, 439, 465, 534. 

DJNZ e, 233. 

Diodos luminosos, 37. 

Diodos luminosos de siete segmen- 
tos (LED), 451-453. 

Direccionamiento, 406, 412. 

Direccionamiento de bits, 416. 

Direccionamiento indirecto de re- 
gistros, 410-411. 

Direccionamiento largo, 416. 

Direccionamiento de registros, 406. 

Direccionamiento, técnicas de, 406. 

Directo, direccionamiento, 408. 

Directorio, 517, 521. 

Directorio de ficheros, 517. 

Directorio de dos niveles, 517. 

Dispositivos múltiples, 476. 

División binaria, 127. 

División de 16 por 8, 126-129. 

División de 8 bits, 131-132. 

DMA, 463, 468. 

Doble precisión, formato de, 31. 

Documentación, 92. 

DOS, 558. 


E, 58. 

EBCDIC, 36. 

Editor, 559. 

EI, 234. 

Ejecución, 53, 66, 576. 
Ejecutar, 68. 

Ejemplos de disefio, 523. 
Elemento aleatorio, 517. 
.Elemento mayor, 500-502. 
Eliminación, 527, 539, 548. 
Eliminación de un elemento, 538. 
Empujar, 50, 71, 152. 
Emulador, 559. 

Emulador interno, 561. 
END, 574. 


ENDC, 577. 

ENDM, 574. 

Ensamblado condicional, 576. 

Ensamblador, 91, 558, 568. 

Entrada/salida, 154, 429, 490. : 
dispositivos de, 483, 495. 
instrucciones de, 179, 430. 

Entrada/salida en paralelo, 45. 

EPROM, 561. 

EQU, 572. 

Error, 561. 

Errores lógicos, 558. 

E/S por zona de memoria, 154. 

Estado, 28, 80, 446, 486. 

Estructuras de datos, 515. 

Ftiqueta, campo de, 566. 


‚ EX AF, АЕ, 235. 


EX DE, HL, 236. 
Exponente, 34-35. 
EX (SP), HL, 237. 
EX (SP), IX, 238. 
EX (SP), IY, 240. 
Extraer, 51, 71, 152. 
EXX, 242. 


F, 58. 

Fallos de alimentación, 45. 
FIFO, 519. 

Filtro, 164, 497. 


H, 58, 173. 

HALT, 86, 182, 243. 
HEX, 500. 

Hexadecimal, 38-40, 451. 


I, 61. 

Identificación de interruptores, 471. 

IFF1, 470. 

IFF2, 470. 

IM 0, 244. 

IM 1, 245. 

IM 2, 246. 

Implícito, direccionamiento, 406, 
412. 

Impresora, 40, 449, 466. 

Impulso, 431, 436. 

IN r (C), 247. 

IN A, (N), 249. 

INC (HL), 252. 

INC г, 250. 

INC rr, 251. 

INC (IX + d), 253. 

INC (IY + d), 255. 

INC IX, 256. 

INC IY, 257. 

Incremento, 160, 409. 
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Incrementador, 54. 
IND, 258. 
Indexación, 61. 
Indexado, direccionamiento, 157, 
409, 414, 517. 
Indirecto, direccionamiento, 410- 
412, 415, 516. 
Indirecto indexado, direccionamien- 
to, 410. 
INDR, 259, 
INI, 260. 
INIR, 262. 
Inmediato, direccionamiento, 103, 
156, 406, 412-413. 
Inserción, 527, 547. 
Inserción de un elemento, 525, 537. 
Instrucción, 91. 
campo, 566. 
formatos, 62. 
registro, 53, 61. 
tipos, 107. 
780, 151. 
Instrucción breve, 17. 
Instrucción condicional, 48. 
Instrucciones automáticas del 780, 
137, 421, 423. 
Instrucciones de control, 154, 181. 
Instrucciones ejecutables, 15. 
Instrucciones especiales para cifras, 
168. 
Instrucciones de intercambio, 159. 
Instrucciones de proceso de datos, 
161. 
INT, 86. 
Interpretado, 65. 
Intérprete, 52, 557. 
Interrupción, 435, 467-476, 478-479. 
modo 0, 471. 
modo 1, 474. 
modo 2, 474. 
Interrupción no filtrable, 468. 
Interrupciones simultáneas, 478. 
Introducción de caracteres, 496. 
ТОКО, 87, 471. 
IR, 53. 
IX, 50, 60. 
ГУ, 60. 


JP cc, pq, 263. 
JP nn, 83. 

JP pq. 265. 
JP (HL), 266. 
JP (IX), 267. 
JP (Гу), 268. 
JR cc, e, 269. 
JR e, 271. 


JUMP (salto), 83, 170, 175, 408. 


L, 60. 

LD A, (nn), 65, 80. 
LD D, C, 68. 
LDD, 160. 

LDDR, 137, 160. 
LDI, 160. 

LDIR, 160. 

LD dd, (nn), 272. 
LD dd, nn, 274. 
LD r, n, 275. 

LD г, r', 63, 276. 
LD (BC) A, 277. 
LD (DE), A, 278. 
LD (HL), n, 279. 
LD (HL), r, 280. 
LD r, (HL), 330. 
LD r, (IX + d), 281. 
LD г, (IY + d), 283. 
LD (IX + d), n, 285. 
LD (IY + d), 287. 
LD (IX + d), r, 289. 
LD (IY + d), г, 291. 
LD (nn) A, 295. 
LD A, (nn), 293. 
LD (nn) dd, 297. 
LD (nn), HL, 299. 
LD (nn), 1X, 301. 
LD (nn), ТУ, 303. 
LD A, (BC), 305. 
LD A, (DE), 306. 
LD A, 1, 307. 

LD A, В, 309. 

LD I, A, 308. 

LD HL (nn), 310. 
LD IX, nn, 312. 
LD IX, (nn), 313. 
LD ТУ, nn, 315. 
LD IY, (nn), 316. 
LD R, A, 318. 

LD SP, HL, 319. 
LD SP, IX, 320. 
LD SP, IY, 321. 


LDIR, 328. 

Lectora de cinta de papel, 465. 

LED, 37, 450. 

LED múltiples, 452. 

Lenguaje ensamblador, 63, 556, 
568. 

Lenguajes de alto nivel, 557. 

Lenguajes de programación, 14. 

LIFO, estructura, 516, 519. 


Lista, 516, 524-525, 533. 

Lista alfabética, 533, 540, 543-544, 

Lista circular, 520. 

Lista doblemente encadenada, 521. 

Lista encadenada, 517, 520, 542, 
544-546, 549. 

Lista secuencial, 516. 

Lista sencilla, 525. 

Listado, 566. 

Literal, 65, 406, 422. 570. 

Lógica, 163, 533. 

Lógica binaria, 16. 

Lógica de decodificación, 46. 

Lógica de interrupciones, 481, 

Lógica sincronizada por reloj, 80. 


Llamada a subrutina, 138, 141. 
Llamadas internas, 140. 


MI, 86. 

MACRO, 574-576. 

Manipulación de bits, 168-169. 

Manipulador de interrupciones, 
466. 

Mantisa, 35. 

Mantisa normalizada, 35. 

Mapa de memoria, 421, 562. 

Mecanismo de subrutina, 138. 

Medio acarreo, bandera de (H), 
173. 

Memoria auxiliar de datos, 483. 

Memoria de lectura-escritura, 45, 
70. 

Memoria de sólo lectura, 45. 

Memorias auxiliares, 58. 

Mensajes de error, 568. 

Microinstrucciones, 80, 

Microordenador monoplaca, 563. 

Mnemotécnico, 63, 556. 

Modelo del programador, 90. 

Modos, 411. 

Modos de direccionamiento, 406- 
407, 412. 

Monitor, 558. 

MOS, tecnología (6502), 419. 

MREQ, 86. 

Multiplexor, 49, 60. 

Multiplicación, 107-116, 146-148. 

Multiplicación de 16 por 16, 124- 
126. 

Multiplicación mejorada, 120-124. 

MUX, 49, 60. 


N, 31. 
NEG, 331. 
Negativo, 21, 23, 29-30. 


Nibble, 16, 33. 

NMI, 85-86, 468. 

NOP, 86, 332. 
Normalizar, 34. 
Notación posicional, 18. 
Número con signo, $08. 


O exclusiva, 28. 

Octal, 38-39. 

Opciones de programación, 555. 
Operación inmediata, 65. 
Operación de lectura, 92. 
Operaciones lógicas, 135-136. 
Operaciones de salto, 166. 
Operando, 95, 97, 406. 
Operandos de almacenamiento, 97. 
OR, 164-165. 

OR s, 333. 

Ordenes, 15. 

ORG, 572. 

Organización, 462. 
Organización hardware, 43. 
OTDR, 335. 

OTIR, 337. 

OUT (С), г, 339. 

OUT (М), A, 340. 

OUTD, 341. 

OUTI, 342. 


Página cero, direccionamiento por, 
408, 413. 

Panel frontal, 40, 565. 

Pantalla, 40, 563. 

Parámetros de subrutinas, 144. 

Pares de registros, 49. 

Paridad/desbordamiento (P/V), 171. 

Paso a paso, 435. 

Pastilla de entrada/salida progra- 
mable, 483. 

Patillas del uP, 85. 

PC, 49, 413, 478. 

Perforadora, 466. 

Pila, 50, 140, 144, 467, 478-479, 515, 
519. 

РТО, 45, 483-490. 

PIO estándar, 483. 

PIO Zilog 780, 488-489. 

POP qq, 343. 

POP IX, 345. 

POP IY, 347. 

Positivo, 23, 29. 

Postindexación, 409. 

Precisión múltiple, 94, 

Preindexación, 409, 

Proceso de datos, 152. 

Producción de paridad, 498. 
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Programa, 15, 46. 

Programa de carga, 559. 

Programa de control, 45. 

Programa de puesta a punto, 559. 

Programación, 14, 15, 488, 493, 579. 

Protección de registros, 471. 

Protegido, 46. 

Puerta, 483, 487-488. 

Puesta a punto, 15. 

Puntero de la lista, 518. 

Puntero de la pila, 516. 

Punteros, 49, 50, 63, 410, 516, 520, 
525, 528. 

Punto de bifurcación, 111. 

Punto de interrupción, 560, 561. 

PUSH аа, 349. 

PUSH IX, 351. 

PUSH IY, 353. 


В, 61. 

КАМ, 46, 73, 500, 563. 

ВО, 87. 

Recursos físicos, 563, 565. 

Recursos lógicos, 558, 563. 

Recurrencia, 143. 

Referencia, programa de, 440. 

Registro destino, 64. 

Registro de dirección, 485, 

Registro de dirección de datos, 484. 

Registro de entrada, 430. 

Registro de estado, 47, 58. 

Registro de índice, 50, 60, 409. 

Registro-interrupción, 180. 

Registro de interrupción-dirección 
de página, 61. 

Registro de refresco de memoria, 
61. 

Registro de salida, 430. 

Registro temporal, 58. 

Registros, 28, 48-49, 144, 408, 444. 

Registros de control, 494-495. 

Registros de direcciones, 49. 
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